首页 > 代码库 > 当泛型方法推断,扩展方法遇到泛型类型in/out时。。。
当泛型方法推断,扩展方法遇到泛型类型in/out时。。。
说到泛型方法,这个是.net 2.0的时候引入的一个重要功能,c#2.0也对此作了非常好的支持,可以不需要显试的声明泛型类型,让编译器自动推断,例如:
1 void F<T>(T value){} 2 //... 3 int i = 0; 4 F(i);
此时,编译器可以自动推导出这里的T就是int,这极大的方便了我们写代码的效率。
说到扩展方法,这个是.net 3.5的时候引入的另一个重要功能,c#3.0也在linq中大量的应用这个功能,当扩展方法是扩展一个泛型的类型时,显然也不需要我们指定具体的泛型类型,编译器会为我们自动推断,例如:
1 static void F<T>(this List<T> list){} 2 //... 3 List<int> list = new List<int>(); 4 list.F();
最后说到协变和逆变(也就是c#中的in/out),这个是.net 4.0的时候引入的一个重要的功能,例如:
1 Func<string> foo = () => "Foo"; 2 Func<object> bar = foo;
然后,我们将泛型方法推断和协变和逆变放在一起:
1 public static void Foo(this Action<string> action){} 2 //... 3 Action<object> action = o => {}; 4 action.Foo();
看起来很不错,不过要是遇到些复杂点的会怎么样?
1 public static void Foo(this IEnumerable<IEnumerable<object>> that) {} 2 //... 3 List<List<string>> bar = new List<List<string>>(); 4 bar.Foo();
看到这里相信所有都为c#的in/out拍手叫好,不过别急,除了out+out我们还可以玩令人抓狂的in+in:
1 public static void Foo(this Action<Action<object>> that) {} 2 //... 3 Action<Action<string>> action = a => a("O_O"); 4 action.Foo();
看到这里有没有发现什么问题?如果你没觉得有什么不舒服的感觉,说明你一定是懂协变和逆变的高手或是完全不懂的初学者。
先想下定义:Action<in T>,T 是in的,也就是Action<object>里面的object可以被string这样更具体的类型替代,而这里Action<Action<string>>里面的Action<string>被Action<object>替代了,怎么看都感觉有些怪异,不过在细细品味一下,就可以发现这个结果是完全合理的。string虽然比object更具体,不过一个接受string的方法可比一个接受object的方法更抽象,所以可以简单的得到一个结论:in+in=>out
文章要是到这里结束,估计很多人就认为本文是对c#的无比赞美了吧,不过,重点是这里,别忘了,多个泛型参数可以玩出很多猥琐的东西,例如,双/多泛型锁定(随便起的名字):
1 public static void Foo<T>(this Action<T, T> that) {} 2 //... 3 Action<string, object> action = (s, o) => {}; 4 action.Foo();
c#编译器对扩展方法支持的确是有一手,这么变态的T也可以被推断出是object,不得不佩服一把,再来看看out的情况(别忘了前面的结论in+in=>out):
1 public static void Foo<T>(this Action<Action<T>, Action<T>> that) {} 2 //... 3 Action<Action<string, object>> action = (s, o) => {}; 4 action.Foo();
c#编译器依然表现出专业的结果,正确的推断出了T应当是string,不过,泛型方法的类型推断却完全是另外一番风景:
1 public void Foo<T>(Action<T, T> that) {} 2 //... 3 Action<string, object> action = (s, o) => {}; 4 Foo(action); // failed. 5 Foo<object>(action); // failed. 6 Foo((Action<object, object>)action); // succeeded.
看到这个结果是不是想大骂c#编译器:这也太山寨了吧。
别急,我们还可以玩得更加浮云:
1 public static Foo<T>(this Action<Action<T>, Action<T>> that) {} 2 // ... 3 Action<Action<List<string>>, Action<ArrayList>> bar = null; 4 bar.Foo(); // failed. 5 bar.Foo<IEnumerable>(); // succeeded.
或者这个:
public static Foo<T>(this Action<T, T> that) {} // ... Action<IEnumerable<char>, IComparable<string>> action = null; action.Foo(); // failed. action.Foo<string>(); // succeeded.
c#编译器显然不想瞎猜T的类型是什么,要求必须编程者明确指出T的具体类型。