Jeffrey Zhao真是神一样的存在,伊太结棍了(上海话),每次看他的博客得使劲使劲使劲地啃。本篇源于Jeffery Zhao的"逆泛型执行器"这篇文章。该文提到了为以下的接口写一个泛型方法:
public interface IRecord{string GetString(string field);int GetInt(string field);long GetLong(string field);}
先来实现该接口:
public class MyRecord : IRecord{public string GetString(string field){return field + "--added string";}public int GetInt(string field){return int.Parse(field + "1");}public long GetLong(string field){return long.Parse(field);}}
通常,在客户端这样调用:
static void Main(string[] args){MyRecord myRecord = new MyRecord();Console.WriteLine(myRecord.GetString("hello"));Console.WriteLine(myRecord.GetInt("1"));Console.WriteLine(myRecord.GetLong("2"));Console.ReadKey();}
以上,对于IRecord接口的各个方法而言,它们处在同一个接口,有相同的方法参数,唯一不同的是返回类型,看来有必要请出泛型了。
对于接口方法来说,接口就是它们的"天",它们得"一辈子"待在这里。而对于泛型而言,它以更高的视角来俯视接口和它的方法们。
可能,我们需要这样一个针对IRecord接口的泛型方法:SomeExtenion.Get<T>(IRecord record, string str)。
这里的T是什么,返回类型就是什么, 如果T是string类型,那就返回string类型,等等。输入参数就IRecord是接口和string类型。
如何做呢?
有这样的一个类的轮廓模模糊糊地出现在了脑海里:
public static class RecordExtensions{public static T Get<T>(IRecord record, string field){}}
可是问题来了,如何返回T类型呢?
Jeffrey Zhao给出了答案!
.NET 泛型的奇妙之处便在于其“动态”及“区分”。“动态”在于它可以于运行时进行具体化(相对于 C++ 里的“静态”),不过目前的问题不涉及这点。而“区分”则意味着不同的具体泛型参数,在 .NET 中都是不同的类型,拥有完全分离的元数据,例如方法表(Method Table),以及静态字段等等。
再次回到问题的本质:接口调用方法1,输入string类型参数,返回某个类型1;接口调用方法2,输入string类型参数,返回某个类型2......这不就是一个委托,这不就是Func<IRecord, string, T>吗?
如果我们能把以下存储到某个地方该多好啊!
(irecord, field) => irecord.GetString(field)(irecord, field) => irecord.GetInt(field)(irecord, field) => irecord.GetLong(field)
当Get<T>调用的时候再把这些泛型委托取出来。
Jeffery Zhao又给出了答案!
一个内部类 Cache,它起到了缓存的作用。
于是,再完善一下RecordExtensions这个类。
public static class RecordExtensions{private static class Cache<T>{//委托类型字段,输入参数是IRecord和string类型,输出参数Tpublic static Func<IRecord, string, T> Get;}//构造函数中就往缓存中存数据static RecordExtensions(){Cache<string>.Get = (irecord, field) => irecord.GetString(field);Cache<int>.Get = (irecord, field) => irecord.GetInt(field);Cache<long>.Get = (irecord, field) => irecord.GetLong(field);}public static T Get<T>(IRecord record, string field){return Cache<T>.Get(record, field);}}
以上,在静态类的静态构造函数中,把委托赋值给静态类中作为缓存的静态类Cache的委托类型的字段Get,由于.NET的“区分”性,缓存中的泛型委托拥有完全分离的元数据,当调用Get<T>方法的时候,会从缓存中取出对应的委托。
最后,在客户端这样调用:
static void Main(string[] args){MyRecord myRecord = new MyRecord();Console.WriteLine(myRecord.GetString("hello"));Console.WriteLine(myRecord.GetInt("1"));Console.WriteLine(myRecord.GetLong("2"));Console.WriteLine(RecordExtensions.Get<string>(myRecord, "hello"));Console.WriteLine(RecordExtensions.Get<int>(myRecord, "1"));Console.WriteLine(RecordExtensions.Get<long>(myRecord, "2"));Console.ReadKey();}
通过调用接口方法和泛型方法,得到的结果是一样一样滴。
Thanks, Jeffrey Zhao~~