泛型在我理解就好比是一個袋子,可以裝下任何東西.我們當然不希望用這個袋子又裝好吃的又要裝垃圾.打一個更形象的比方,去超市買一包餅乾,當你結帳的時候服務員會給你一個袋子裝這包餅乾.但是對於我們來說餅乾與餅乾的封裝袋都裝到了這個大袋子裡,而餅乾是我們所需要的,封裝袋子就是垃圾.這時就會引出一個概念,"裝箱、拆箱".將封裝袋裝餅乾的過程就叫"裝箱",而我們拆封裝吃餅乾的過程叫"拆箱".顯然這對於吃餅乾來說是非常浪費時間的.所以我們可能會需要一個可以做自我提示的袋子,也就是說如果你想用這個袋子來裝餅乾,那麼你用它來裝糖果,它就會自動提示你,它已經被你指定成裝餅乾了.
上面的例子雖然不太貼切,但是對於理解泛型還是有些協助的.
大概有以下幾個集合類型:
1. List,這是我們應用最多的泛型種類,它對應ArrayList集合。
2. Dictionary,這也是我們平時運用比較多的泛型種類,對應Hashtable集合。
3. Collection對應於CollectionBase
4. ReadOnlyCollection 對應於ReadOnlyCollectionBase,這是一個唯讀集合。
5. Queue,Stack和SortedList,它們分別對應於與它們同名的非泛型類。
看一下這個類:
PersonCollection
1public class PersonCollection : IList
2{
3 private ArrayList _Persons = new ArrayList();
4 public Person this[int index]
5 { get { return (Person) _Persons[index]; } }
6 public int Add(Person item)
7 {
8 _Persons.Add(item);
9 return _Persons.Count - 1;
10 }
11 public void Remove(Person item)
12 { _Persons.Remove(item); }
13
14 object IList.this[int index]
15 {
16 get { return _Persons[index]; }
17 set { _Persons[index] = (Person)value; }
18 }
19 int IList.Add(object item)
20 { return Add((Person)item); }
21 void IList.Remove(object item)
22 { Remove((Person)item); }
23
24
25}
26
27
這個類是Person類的操作類,可以自由的增加或刪除Person類.如果現在要為其他的類寫一個功能與這個類一樣的操作類,相信只需要將Person類替換一下就可以了.不過在瞭解泛型之後你就可以這樣來用.
List<Person> persons = new List<Person>();
persons.Add(new Person());
Person person = persons[0];
比如,如果要將Person類換成Employee類,只需要像這樣寫.
List<Employee> employees= new List<Employee>();
employees.Add(new Employee());
Employee employee= employees[0];
List是C#中已經定義好的泛型類,現在我們可以自己定義它.
TypeHelper
1public class TypeHelper<T>
2
3{
4
5 public String GetType(T t){
6
7 return "Type is "+t.GetType().toString();
8 }
9}
10
11
這裡的T只是一個類型的預留位置,在實際應用的時候,將實際的類型替換掉T就可以.
TypeHelper<Person> typeHelper = new TypeHelper();
typeHelper.GetType(Person);
注意T只是一個預留位置,實際上換上任何符號都可以,千萬不要任為只有T可以做預留位置.
有的時候我們必須要約束一下實際的類型,比如以下的泛型類.
CollectionHelper
1public class CollectionHelper<T,V>
2
3{
4
5 private T Collec = new T();
6
7 public int IndexOf(V v){
8
9 return Collec.IndexOf(v);
10
11 }
12}
13
泛型類中的T,V顯示不是隨便什麼類型都可以代替的,首先這個類型T必須具有IndexOf方法,V必須是一個參考型別.所以這個類要修改一下.
CollectionHelper
1public class CollectionHelper<T,V> where T:IList
2
3 where V:class
4
5{
6
7 private T Collec = new List();
8
9 public int IndexOf(V v){
10
11 return Collec.IndexOf(v);
12
13 }
14
15}
16
where 是關鍵字,T是表示所要進行約束的類型.IList是表示要實現的介面,顯示只要實現IList介面,就一定有IndexOf方法,V的約束是必須是一個類.
如果要求類型必須是一個實值型別的參數,就需要使用struct.如果還需要將類型重新執行個體化就需要使用new()來進行限制,說明該類型必須要有一個無參的建構函式.
注意如果一個類型需要有多個約束進行限制,那麼new()必須寫在最後面.