کلاس های جنریک (عمومی) کلاس هایی هستند که از آنها می توان برای انواع داده های متفاوت استفاده کرد.
قبل از این که به بحث کلاس های عمومی بپردازیم بهتر است به توابع عمومی (جنریک) نگاهی بیندازیم. به یک مثال توجه کنید. فرض کنید می خواهیم یک تابع به نام sum برای جمع کردن دو عدد ورودی بنویسیم. توابع زیر می توانند برای این منظور تعریف شوند:
01.
public
int
sum(
int
a,
int
b)
02.
{
03.
return
(a + b);
04.
}
05.
06.
public
double
sum(
double
a,
double
b)
07.
{
08.
return
(a + b);
09.
}
10.
11.
public
byte
sum(
byte
a,
byte
b)
12.
{
13.
return
(a + b);
14.
}
وقتی که این سه تابع را تعریف می کنیم، هنگامی که تابع sum را با پارامترهایی از نوع int فراخوانی می کنیم، تابع اول فراخوانی می شود. اگر همین تابع sum را با پارامترهایی از نوع double فراخوانی کنیم، تابع دوم فراخوانی می شود و ... . در اینجا ما سه بار تابع sum را تعریف کرده ایم.
در توابع عمومی (جنریک) می توان تابع فوق را یک بار و به صورت زیر تعریف کرد:
01.
public
double
sum(T a, T b)
02.
{
03.
if
(
typeof
(T) ==
typeof
(
int
))
04.
{
05.
return
(
double
)( Convert.ToInt32(a) + Convert.ToInt32(b));
06.
}
07.
if
(
typeof
(T) ==
typeof
(
double
))
08.
{
09.
return
Convert.ToDouble(a) + Convert.ToDouble(b);
10.
}
11.
if
(
typeof
(T) ==
typeof
(
byte
))
12.
{
13.
return
(
double
)( Convert.ToByte(a) + Convert.ToByte(b));
14.
}
15.
return
0;
16.
}
حالا برای فراخوانی تابع sum می توانیم از دستور زیر استفاده کنیم:
1.
double
result = sum(12.509, 49.4);
همانطور که می بینید با توجه به ورودی تابع که بین < >قرار گرفته است، پارامترهای تابع متناسب با آن عمل می کنند. در این مثال شاید دو سوال ذهن شما را درگیر کرده باشد: 1- کد تابع sum نسبت به قبل پیچیده تر شده است. 2- خروجی برای تمام ورودی ها از نوع double است.
در پاسخ به این سوالات باید گفت که با توجه به مثالی که ذکر شده و با توجه به عملی که در این تابع انجام می شود (یعنی عمل جمع) کد تابع کمی پیچیده شده است ولی در مورد توابعی که در ادامه خواهیم گفت، فوق العاده کار را ساده می کنند. و اما در مورد سوال دوم. خروجی تابع sum می تواند از نوع T باشد. یعنی اپر کاربر هنگام فراخوانی تابع sum به صورت sum(12, 20); ž عمل کند خروجی نیز از نوع short بشود ولی در این مثال فقط برای سادگی کار این عمل را انجام ندادیم.
در مورد کلاس های جنریک هم قضیه به همین صورت است. مثال زیر یک کلاس جنریک را که برای گره لیست پیوندی نوشته شده است را نشان می دهد:
01.
public
class
Node
02.
{
03.
public
T data;
04.
public
T next;
05.
06.
public
Node(T value)
07.
{
08.
data = value;
09.
next =
null
;
10.
}
11.
}
اگر همین کلاس را می خواستیم برای انواع داده مختلف تعریف کنیم، کار بسیار دشواری پیش رو داشتیم. اگر برای همین کلاس بخواهیم تعیین کنیم که ورودی کلاس (یعنی T) از نوع داده های عددی باشد، یعنی int, double, byte, short, single و ... می توانیم خط اول کلاس را به صورت زیر تغییر دهیم:
1.
public
class
Node where T:
struct
2.
{
3.
...
4.
}
خط اول تعریف بالا به کامپایلر سی شارپ می گوید که ورودی تعیین شده (یعنی T) فقط باید از انواع داده ای عددی (یا به طور تخصصی تر انواع داده هایی که valued-type هستند و نه referenced-type) باشد.
برای توضیح بیش تر به کلاس لیست پیوندی تعریف شده زیر توجه کنید (توضیحات به خود شما واگذار می شود!):
01.
public
class
Node where T :
struct
02.
{
03.
public
T data;
04.
public
Node next;
05.
public
Node(T value)
06.
{
07.
data = value;
08.
next =
null
;
09.
}
10.
}
11.
12.
public
class
myLinkedList where Y :
struct
13.
{
14.
private
Node head, cur, end;
15.
public
int
Count;
16.
17.
public
myLinkedList()
18.
{
19.
20.
head = cur = end =
null
;
21.
22.
Count = 0;
23.
}
24.
25.
public
void
Add(Y newValue)
26.
{
27.
28.
if
(head ==
null
)
29.
{
30.
head = end =
new
Node(newValue);
31.
}
32.
else
33.
{
34.
end.next =
new
Node(newValue);
35.
end = end.next;
36.
}
37.
Count++;
38.
}
39.
40.
public
Y[] ToArray()
41.
{
42.
Y[] result =
new
Y[Count];
43.
cur = head;
44.
int
i = 0;
45.
while
(cur !=
null
)
46.
{
47.
result[i] = cur.data;
48.
cur = cur.next;
49.
}
50.
return
result;
51.
}
52.
}