.NET 泛型中的逆变 和 协变
逆变和协变是.net 4.0 版本中推出了的概念, 只能在泛型委托 和 泛型接口中使用,
当我们在给泛型变量赋值的时候, 如果赋值表达式看着很安全和谐就是协变, 反之就是逆变
1. 协变 (out)
我们都知道泛型接口的泛型参数不一样时, 默认情况下是不能赋值的, 就算参数存在继承关系也是不行的
如下:
1 Interface1<string> obj1 = null; 2 3 Interface1<object> obj2 = obj1 4 5 // 这个赋值操作是不允许的, obj1 和 obj2是两个完全不相同的对象
如果接口的泛型参数标注为协变了, 那么就可以使表达式成立, 前提泛型参数之间存在继承关系
如下:
1 // 定义接口 2 3 Interface Interface1<out T>{ 4 5 } 6 7 // 在这种情况下表达式是成立的 8 9 Interface1<string> obj1 = null; 10 11 Interface1<object> obj2 = obj1
接口的泛型参数标注为协变, 那么在接口定义的时候 这个泛型参数就不能作为入参传入, 至于原因如下
1 // 定义接口 2 3 Interface Interface1<out T>{ 4 5 } 6 7 8 9 Interface1<string> obj1 = null; 10 11 Interface1<object> obj2 = obj1 12 13 14 15 // 如果我们在Interface1 接口中定义了一个 void Add<T>(T obj) 方法 16 17 // 那么 obj2.Add(123) 这个语句就可以成立, 但是obj2 实际上指向是一个Interface1<string> 对象, Add 方法也只能接收一个string 的值, 那个这个方法的调用就不是安全的了, 会出现异常 18 19 // 所以微软的协变使用out 作为关键字也是在说out 标注的参数只能用于返回不能用于输入
逆变(in)
逆变的定义就是字面意思 很违和的看着不安全的变化, 那么逆变我们在什么时候使用呢, 逆变的应用场景一般就是我们使用委托的时候用
1 // 如我们定义了一个委托 2 3 delegate void DelegateTest <T>(T obj); 4 5 // 定位委托变量 6 DelegateTest<object> dt1 = obj =>{}; 7 8 // 委托DelegateTest 只有入参没有返回值 9 // 那么我们可不可以进行下列操作呢 10 11 DelegateTest<string> dt2 = dt1 12 13 // 按道理这个操作是安全的 dt1 接受一个object 类型的入参, 我们使 dt2 = dt1 那么我们调用委托的时候 dt2("abc") "abc" 符合dt1的入参类型 object 才对的, 为什么会不行呢 14 // 这是因为 dt1 和 dt2 是两个完全不相同的委托, 和 案例1 中的情况是一样的 15 16 // 如果我们定义委托的时候T标注为逆变这个赋值表达式就可以成立了 17 delegate void DelegateTest <in T>(T obj); 18 // 标注逆变的作用就是告诉编译器, 我这个参数只会作为入参, 不会作为返回值, 起约束作用
同时使用逆变和协变的案例:
// 我们C# 中的 系统委托 Func 就是同时使用逆变和协变的 // Func 委托的定义 public delegate TResult Func<in T, out TResult>(T arg); Func<object, string> fuc1 = obj => obj.ToString(); Func<string, object> fuc2 = fuc1; string obj1 = fuc1(123); object obj2 = fuc2("abc");