一、泛型的作用
泛型是.net中十分常见的一种特性。它是在.net 2.0的时候加入。那为什么要在.net 2.0的时候加入泛型这个特性呢?我们首先来看一段代码。
using System;namespace 泛型{class Program{static void Main(string[] args){int iParameter = 1;string sParameter = "hello world";DateTime dtParameter = DateTime.Now;Console.WriteLine("***************************最开始的写法*********************************");CommonMethod.ShowInt(iParameter);CommonMethod.ShowString(sParameter);CommonMethod.ShowDateTime(dtParameter);Console.ReadLine();}}}
using System;using System.Collections.Generic;using System.Text;namespace 泛型{public class CommonMethod{public static void ShowInt(int iParameter){Console.WriteLine(string.Format("这个方法是:{0},它的参数是{1},参数的类型是{2}", typeof(CommonMethod), iParameter.ToString(), iParameter.GetType().ToString()));}public static void ShowString(string sParameter){Console.WriteLine(string.Format("这个方法是:{0},它的参数是{1},参数的类型是{2}", typeof(CommonMethod), sParameter.ToString(), sParameter.GetType().ToString()));}public static void ShowDateTime(DateTime dtParameter){Console.WriteLine(string.Format("这个方法是:{0},它的参数是{1},参数的类型是{2}",typeof(CommonMethod), dtParameter.ToString(), dtParameter.GetType().ToString()));}}}
通过代码可以理解,我们只是要根据不同的类型,输出相同的内容。因为.net 1.0的时候没有泛型的概念,那么我们最简单的做法就是有多少类型,我们就写多少个方法,进行输出。
这样就会造成一些问题:
1、代码量大。.net有那么多类型,每一个类型都写一个方法,那么代码势必会很长。
2、不利于代码维护。当需要对输出内容进行修改时,则每一个方法都需要进行修改。这就是一种灾难。
那么有没有方法解决这些问题呢?
在.net 1.0的时候,我们可以通过这种方式进行解决。先上代码:
using System;namespace 泛型{class Program{static void Main(string[] args){int iParameter = 1;string sParameter = "hello world";DateTime dtParameter = DateTime.Now;Object oParameter = "Object";Console.WriteLine("***************************.net 1.0 最开始的写法*********************************");CommonMethod.ShowInt(iParameter);CommonMethod.ShowString(sParameter);CommonMethod.ShowDateTime(dtParameter);Console.WriteLine("***************************.net 1.0 优化的写法*********************************");CommonMethod.ShowObject(iParameter);CommonMethod.ShowObject(sParameter);CommonMethod.ShowObject(dtParameter);CommonMethod.ShowObject(oParameter);Console.ReadLine();}}}
using System;using System.Collections.Generic;using System.Text;namespace 泛型{public class CommonMethod{public static void ShowInt(int iParameter){Console.WriteLine(string.Format("这个方法是:{0},它的参数是{1},参数的类型是{2}", typeof(CommonMethod), iParameter.ToString(), iParameter.GetType().ToString()));}public static void ShowString(string sParameter){Console.WriteLine(string.Format("这个方法是:{0},它的参数是{1},参数的类型是{2}", typeof(CommonMethod), sParameter.ToString(), sParameter.GetType().ToString()));}public static void ShowDateTime(DateTime dtParameter){Console.WriteLine(string.Format("这个方法是:{0},它的参数是{1},参数的类型是{2}",typeof(CommonMethod), dtParameter.ToString(), dtParameter.GetType().ToString()));}public static void ShowObject(Object oParameter){Console.WriteLine(string.Format("这个方法是:{0},它的参数是{1},参数的类型是{2}",typeof(CommonMethod), oParameter.ToString(), oParameter.GetType().ToString()));}}}
代码中添加了ShowObject方法,参数传递Object类型,然后所有的类型都调用这个方法。那么结果呢?
运行结果是正常的。那么为什么可以这样?
1、任何父类出现的地方都可以用子类代替。
2、Object是一切类型的父类
感觉可以使用这个方法解决之前不同类型,输出相同内容的问题了。为什么后来又出现了泛型呢?
因为使用Object的方法,有两个问题:
1、装箱和拆箱的性能损耗。object是引用类型,当传递值类型时,需要转换成引用类型。
传入一个Int值(栈)需要把值从栈Copy堆里面
2、类型安全问题。当你需要传递一个int类型时,你传递了一个String类型。编译是能通过的,只有等到运行时才会报错。这是不安全的。
总结起来在.net 1.0的时代,主要有几个问题:
1、代码重复比较严重。不同的类型需要写不同的方法,造成代码冗余严重。
2、使用Object参数的形式能解决代码冗余的问题,但是性能上付出了代价。
3、使用Object参数的形式,参数类型是不安全的。任何类型的参数都能传递到方法里。
所以,为了解决这些问题,在.net 2.0的时候就引入了泛型这个特性。
还是一样,先上代码:
using System;namespace 泛型{class Program{static void Main(string[] args){int iParameter = 1;string sParameter = "hello world";DateTime dtParameter = DateTime.Now;Object oParameter = "Object";Console.WriteLine("***************************.net 1.0 最开始的写法*********************************");CommonMethod.ShowInt(iParameter);CommonMethod.ShowString(sParameter);CommonMethod.ShowDateTime(dtParameter);Console.WriteLine("***************************.net 1.0 优化的写法*********************************");CommonMethod.ShowObject(iParameter);CommonMethod.ShowObject(sParameter);CommonMethod.ShowObject(dtParameter);CommonMethod.ShowObject(oParameter);Console.WriteLine("***************************.net 泛型*********************************");CommonMethod.Show<int>(iParameter);CommonMethod.Show<string>(sParameter);CommonMethod.Show<DateTime>(dtParameter);CommonMethod.Show<Object>(oParameter);Console.ReadLine();}}}
using System;using System.Collections.Generic;using System.Text;namespace 泛型{public class CommonMethod{public static void ShowInt(int iParameter){Console.WriteLine(string.Format("这个方法是:{0},它的参数是{1},参数的类型是{2}", typeof(CommonMethod), iParameter.ToString(), iParameter.GetType().ToString()));}public static void ShowString(string sParameter){Console.WriteLine(string.Format("这个方法是:{0},它的参数是{1},参数的类型是{2}", typeof(CommonMethod), sParameter.ToString(), sParameter.GetType().ToString()));}public static void ShowDateTime(DateTime dtParameter){Console.WriteLine(string.Format("这个方法是:{0},它的参数是{1},参数的类型是{2}", typeof(CommonMethod), dtParameter.ToString(), dtParameter.GetType().ToString()));}public static void ShowObject(Object oParameter){Console.WriteLine(string.Format("这个方法是:{0},它的参数是{1},参数的类型是{2}", typeof(CommonMethod), oParameter.ToString(), oParameter.GetType().ToString()));}public static void Show<T>(T tParameter){Console.WriteLine(string.Format("这个方法是:{0},它的参数是{1},参数的类型是{2}", typeof(CommonMethod), tParameter.ToString(), tParameter.GetType().ToString()));}}}
注意里面Show的方法,这个就是泛型方法的定义。那么运行结果呢?请看图:
可以看到是没有问题的。
泛型的作用是什么?我用一句话概括就是不同类型的参数进行相同的操作时,你就可以考虑使用泛型。
二、泛型的定义
public static void Show<T>(T tParameter){Console.WriteLine(string.Format("这个方法是:{0},它的参数是{1},参数的类型是{2}", typeof(CommonMethod), tParameter.ToString(), tParameter.GetType().ToString()));}
泛型方法与一般方法的区别在方法名后有一对<>。其中T表示的是一个占位符,意思是我这里会指定一个类型,但是我现在不知道,我把位置给占着。等要使用到这个方法时,我在指定类型。其采用的思想延迟思想。
其中占位符可以是一个,也可以是多个。比如Show<T,N,M,Two>都行。但是请注意,这里的占位符不能是关键词。比如class,static等。
三、泛型有没有解决问题?
第一个问题,代码冗余,这个比较明显,在main方法里面都调用了一个方法,代码简洁不少。
第二个性能问题,那我们就写一个测试代码。
首先添加一个类Monitor。代码如下:
using System;using System.Collections.Generic;using System.Diagnostics;using System.Text;namespace 泛型{public class Monitor{public static void Show(){Console.WriteLine("****************Monitor******************");{int iValue = 12345;long commonSecond = 0;long objectSecond = 0;long genericSecond = 0;{Stopwatch watch = new Stopwatch();watch.Start();for (int i = 0; i < 100000000; i++){ShowInt(iValue);}watch.Stop();commonSecond = watch.ElapsedMilliseconds;}{Stopwatch watch = new Stopwatch();watch.Start();for (int i = 0; i < 100000000; i++){ShowObject(iValue);}watch.Stop();objectSecond = watch.ElapsedMilliseconds;}{Stopwatch watch = new Stopwatch();watch.Start();for (int i = 0; i < 100000000; i++){Show<int>(iValue);}watch.Stop();genericSecond = watch.ElapsedMilliseconds;}Console.WriteLine("一般方法耗时={0}毫秒,Object方法耗时={1}毫秒,泛型方法耗时={2}毫秒" , commonSecond, objectSecond, genericSecond);}}#region PrivateMethodprivate static void ShowInt(int iParameter){//do nothing}private static void ShowObject(object oParameter){//do nothing}private static void Show<T>(T tParameter){//do nothing}#endregion}}
main方法:
using System;namespace 泛型{class Program{static void Main(string[] args){for (int i = 0; i < 5; i++){Monitor.Show();}Console.ReadLine();}}}
Monitor类中定义了ShowInt,ShowObject,Show三类方法,每个方法内都是空的,没有做任何操作。然后在Show方法中,调用ShowInt,ShowObject,Show各100000000次,通过Stopwatch计时器计算每个方法执行10000000次所需要的时间(毫秒)。
Main方法中,为了比较全面的看到时间,我这里调用了5次Show方法。结果如图:
明显的能看出,Object方法耗时最长,一般方法的耗时和泛型的耗时基本相差无几。
所以在性能上的问题,我们也能得到了解决。
第三个问题:类型安全问题。
当我们使用泛型方法时,在调用的适合就需要指定类型。
int iValue = 12345;
string sValue=”Hello World”;
Show(iValue)
这样是可以成功的,但是如果我们写成这样:
Show(sValue)
则编译不通过,从一定程度上解决了类型安全的问题。
四、泛型缓存
为什么泛型的效率这么高呢?这里面涉及到泛型缓存的概念。
类中的静态类型无论实例化多少次,在内存中只会有一个。静态构造函数只会执行一次。在泛型类中,T类型不同,每个不同的T类型,都会产生一个不同的副本,所以会产生不同的静态属性、不同的静态构造函数。简单的说就是不同类型的T会生成相同的副本,而相同类型的T会只用同一个副本。
请看下面的例子:
GenericCache类:
using System;using System.Collections.Generic;using System.Text;namespace 泛型{public class GenericCache<T>{static GenericCache(){Console.WriteLine("这是 GenericCache 静态构造函数");_TypeTime = string.Format("{0}_{1}", typeof(T).FullName, DateTime.Now.ToString("yyyyMMddHHmmss.fff"));}private static string _TypeTime = "";public static string GetCache(){return _TypeTime;}}}
GenericCacheTest类:
using System;using System.Collections.Generic;using System.Text;using System.Threading;namespace 泛型{public class GenericCacheTest{public static void Show(){for (int i = 0; i < 5; i++){Console.WriteLine("这是第{0}次执行......", (i + 1).ToString());Console.WriteLine(GenericCache<int>.GetCache());Thread.Sleep(100);Console.WriteLine(GenericCache<long>.GetCache());Thread.Sleep(100);Console.WriteLine(GenericCache<DateTime>.GetCache());Thread.Sleep(100);Console.WriteLine(GenericCache<string>.GetCache());Thread.Sleep(100);Console.WriteLine(GenericCache<GenericCacheTest>.GetCache());Thread.Sleep(100);}}}}
main方法:
using System;namespace 泛型{class Program{static void Main(string[] args){GenericCacheTest.Show();Console.ReadLine();}}}
运行后结果为:
观察结果可以看到,5次执行相同类型的时间都是一致的。并且从第二次开始,执行时并没有进入构造函数。利用泛型的这一个特性,可以实现缓存。
注意:只能为不同的类型缓存一次。泛型缓存比字典缓存效率高。泛型缓存不能主动释放