0

کلاس های عمومی (جنریک) - Generic Classes

 
hosseinb68
hosseinb68
کاربر طلایی1
تاریخ عضویت : بهمن 1389 
تعداد پست ها : 1269
محل سکونت : بوشهر

کلاس های عمومی (جنریک) - Generic Classes
دوشنبه 6 آذر 1391  8:22 PM

کلاس های جنریک (عمومی) کلاس هایی هستند که از آنها می توان برای انواع داده های متفاوت استفاده کرد.

قبل از این که به بحث کلاس های عمومی بپردازیم  بهتر است به توابع عمومی (جنریک) نگاهی بیندازیم. به یک مثال توجه کنید. فرض کنید می خواهیم یک تابع به نام sum برای جمع کردن دو عدد ورودی بنویسیم. توابع زیر می توانند برای این منظور تعریف شوند:

view sourceprint?
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 را تعریف کرده ایم.

در توابع عمومی (جنریک) می توان تابع فوق را یک بار و به صورت زیر تعریف کرد:

view sourceprint?
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 می توانیم از دستور زیر استفاده کنیم:

view sourceprint?
1.double result = sum(12.509, 49.4);

همانطور که می بینید با توجه به ورودی تابع که بین < >قرار گرفته است، پارامترهای تابع متناسب با آن عمل می کنند. در این مثال شاید دو سوال ذهن شما را درگیر کرده باشد: 1- کد تابع sum نسبت به قبل پیچیده تر شده است. 2- خروجی برای تمام ورودی ها از نوع double است.
در پاسخ به این سوالات باید گفت که با توجه به مثالی که ذکر شده و با توجه به عملی که در این تابع انجام می شود (یعنی عمل جمع) کد تابع کمی پیچیده شده است ولی در مورد توابعی که در ادامه خواهیم گفت، فوق العاده کار را ساده می کنند. و اما در مورد سوال دوم. خروجی تابع sum می تواند از نوع T باشد. یعنی اپر کاربر هنگام فراخوانی تابع sum به صورت sum(12, 20); ž عمل کند خروجی نیز از نوع short بشود ولی در این مثال فقط برای سادگی کار این عمل را انجام ندادیم.

در مورد کلاس های جنریک هم قضیه به همین صورت است. مثال زیر یک کلاس جنریک را که برای گره لیست پیوندی نوشته شده است را نشان می دهد:

view sourceprint?
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 و ... می توانیم خط اول کلاس را به صورت زیر تغییر دهیم:

view sourceprint?
1.public class Node where T: struct
2.{
3....
4.}

خط اول تعریف بالا به کامپایلر سی شارپ می گوید که ورودی تعیین شده (یعنی T) فقط باید از انواع داده ای عددی (یا به طور تخصصی تر انواع داده هایی که valued-type هستند و نه referenced-type) باشد.

برای توضیح بیش تر به کلاس لیست پیوندی تعریف شده زیر توجه کنید (توضیحات به خود شما واگذار می شود!):

view sourceprint?
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.        // set head, cur and end to null
20.        head = cur = end = null;
21.        // count contains count of items available in the list
22.        Count = 0;
23.    }
24. 
25.    public void Add(Y newValue)
26.    {
27.        // if there is not any data create head of list
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.}

 

تشکرات از این پست
دسترسی سریع به انجمن ها