泛型
对泛型的理解进行梳理,并将之变成自己的东西,再加之自己的理解想陈述的更加简单直观些,能力有限,且文笔功底略显不足,因此如有欠妥之处请不吝赐教.
泛型如何理解?
1.实现代码复用,无须对不同类型编写近乎相同的代码(仅仅参数类型不同)。而实际上Console.Write()方法也并未采用泛型,因此出现下图若干重载函数,至于为什么未采用泛型在此不做研究。
2.使用了泛型的集合可以保证其类型安全以及可以避免集合添加元素、取出元素时候的装箱、拆箱操作。
一个简单的例子
初学泛型曾看了一些文章,好像大都直接拿泛型集合说事,本人比较不够聪明,总以为泛型仅仅用于泛型集合的,以至于很长时间以来一想到泛型就想到泛型集合。如果有园友初学泛型,那么暂且不要关心泛型集合,只关心泛型,且看如下例子。
我们需要将两个值ToString()后再相加并输出到屏幕,那么我们可能会像如下这么写(可能这么简单的功能没人会像如下这么做,大概会只封装一个以string为参数的函数,在传参的时候ToString(),此处只是为了举例的方便,假设函数没那么简单的情况下):
class MyHelper { public static void AddAndPrint( int i, int j) { Console.WriteLine( string .Format( "The value is {0}" , i.ToString() + "@" + j.ToString())); } public static void AddAndPrint( float i, float j) { Console.WriteLine( string .Format( "The value is {0}" , i.ToString() + "@" + j.ToString())); } public static void AddAndPrint( double i, double j) { Console.WriteLine( string .Format( "The value is {0}" , i.ToString() + "@" + j.ToString())); } } |
请注意,上面三个AddAndPrint函数除了参数类型以外,函数体内部实现完全一样。那么我们会这样调用它们:
static void Main( string [] args) { MyHelper.AddAndPrint(100, 120); MyHelper.AddAndPrint(100.123F, 120F); MyHelper.AddAndPrint(100.456D, 120.666D); Console.ReadKey(); } |
那么很容易就可以想到,AddAndPrint函数仅仅参数类型不一样却“不得不”写了三次(如果是更多次呢),这显然是不能接受的。这时候我们可以通过泛型来进行代码重用。三个函数参数类型分别是int、float、double,那么我们合并三个函数为一个函数,参数类型就不能是任一确定类型,我们可以假定这个类型是T(取自Type首字母,当然.net许多类库的代码会使用诸如TEntity、TElement之类的替代符可能更容易理解),那么我们可以将如上三个函数进行合并,如下:
public static void AddAndPrint(T t1, T t2) { Console.WriteLine( string .Format( "The value is {0}" , t1.ToString() + "@" + t2.ToString())); } |
那么可以通过如下方式将类型传入(因为是静态函数,因此每次调用的时候在类名后面以如<int>的形式指定其类型,如果调用非静态函数,那么在调用构造函数的时候指定即可):
class MyHelper<T> { ... } |
注意,MyHelper<T>与AddAndPrint(T t1, T t2)须一致,当然也可以有多种类型,如MyHelper<T1,T2>与AddAndPrint(T1 t1,T2 t2),完整代码如下:
class MyHelper<T> { public static void AddAndPrint(T t1, T t2) { Console.WriteLine( string .Format( "The value is {0}" , t1.ToString() + "@" + t2.ToString())); } } static void Main( string [] args) { MyHelper.AddAndPrint(100, 120); MyHelper.AddAndPrint(100.123F, 120F); MyHelperAddAndPrint(100.456D, 120.666D); Console.ReadKey(); } |
泛型集合简单例子
其实泛型集合不过是在编写集合类(以及其节点Node类)时采用泛型而已。下面就以一个简单的单向链表为例。
MyNode类,不指定Data的类型,以占位符T代替:
class MyNode<T> { public MyNode(T t) { this .Data = t; } public T Data { get ; set ; } public MyNode<T> Next { get ; set ; } public override string ToString() { return Data.ToString(); } } |
那么我们的MyList类可以像下面这样,仅仅简单实现添加功能:
class MyList<T> { public MyNode<T> Head { get ; set ; } public MyList(T t) { Head = new MyNode<T>(t); } public void Append(T t) { MyNode<T> temp = this .Head; while (temp.Next != null ) { temp = temp.Next; } MyNode<T> newNode = new MyNode<T>(t); temp.Next = newNode; } public override string ToString() { StringBuilder sb = new StringBuilder(); MyNode<T> temp = this .Head; int index = 0; do { sb.Append( string .Format( "The {0} Element is {1}\n" , index, temp.Data.ToString())); index++; temp = temp.Next; } while (temp != null ); return sb.ToString(); } } |
调用:
static void Main( string [] args) { MyList< string > myList = new MyList< string >( "first" ); myList.Append( "second" ); MyList< int > mylist2 = new MyList< int >(12); mylist2.Append(1231); Console.WriteLine(mylist2.ToString()); Console.ReadKey(); } |