“协变”是指能够使用与原始指定的派生类型相比,派生程度更大的类型。
“逆变”则是指能够使用派生程度更小的类型。
解释的很正确,大致就是这样,不过不够直白。
直白的理解:
“协变”->”和谐的变”->”很自然的变化”->string->object :协变。
“逆变”->”逆常的变”->”不正常的变化”->object->string 逆变。
上面是个人对协变和逆变的理解,比起记住那些派生,类型,原始指定,更大,更小之类的词语,个人认为要容易点。
下面是一则笑话:
一个星期的每一天应该这样念:
星期一 = 忙day;
星期二 = 求死day;
星期三 = 未死day;
星期四 = 受死day;
星期五 = 福来day;
星期六 = 洒脱day;
星期天 = 伤day
为了演示协变和逆变,以及之间的区别,请创建控制台程序CAStudy,手动添加两个类:
因为是演示,所以都是个空类,
只是有一点记住Dog 继承自Animal,
所以Dog变成Animal 就是和谐的变化(协变),而如果Animal 变成Dog就是不正常的变化(逆变)
在Main函数中输入:
因为Dog继承自Animal,所以Animal aAnimal = aDog; aDog 会隐式的转变为Animal.
但是List<Dog> 不继承List<Animal> 所以出现下面的提示:
如果想要转换的话,应该使用下面的代码:
List<Animal> lstAnimal2 = lstDogs.Select(d => (Animal)d).ToList();
可以看到一个lstDogs 变成lstAnimal 是多么复杂的操作了。
正因如此,所以微软新增了两个关键字:Out,In,下面是他们的msdn解释:
协变的英文是:“covariant”,逆变的英文是:“Contravariant”
为什么Microsoft选择的是”Out” 和”In” 作为特性而不是它们呢?
我个人的理解:
因为协变和逆变的英文太复杂了,并没有体现协变和逆变的不同,但是out 和 in 却很直白。
out: 输出(作为结果),in:输入(作为参数)
所以如果有一个泛型参数标记为out,则代表它是用来输出的,只能作为结果返回,而如果有一个泛型参数标记为in,则代表它是用来输入的,也就是它只能作为参数。
目前out 和in 关键字只能在接口和委托中使用,微软使用out 和 in 标记的接口和委托大致如下:
先看下第一个IEnumerable<T>
和刚开始说的一样,T 用out 标记,所以T代表了输出,也就是只能作为结果返回。
public static void Main()
{
Dog aDog = new Dog();
Animal aAnimal = aDog;
List<Dog> lstDogs = new List<Dog>();
//List<Animal> lstAnimal = lstDogs;
List<Animal> lstAnimal2 = lstDogs.Select(d => (Animal)d).ToList();
IEnumerable<Dog> someDogs = new List<Dog>();
IEnumerable<Animal> someAnimals = someDogs;
}
因为T只能做结果返回,所以T不会被修改, 编译器就可以推断下面的语句强制转换合法,所以
IEnumerable<Animal> someAnimals = someDogs;
可以通过编译器的检查,反编译代码如下:
虽然通过了C#编译器的检查,但是il 并不知道协变和逆变,还是得乖乖的强制转换。
在这里我看到了这句话:
IEnumerable<Animal> enumerable2 = (IEnumerable<Animal>) enumerable1;
那么是不是可以List<Animal> lstAnimal3 = (List<Animal>)lstDogs; 呢?
想要回答这个问题需要在回头看看Clr via C# 关于泛型和接口的章节了,我就不解释了,
答案是不可以。
上面演示的是协变,接下来要演示下逆变。
为了演示逆变,那么就要找个in标记的接口或者委托了,最简单的就是:
在Main函数中添加:
Action<Animal> actionAnimal = new Action<Animal>(a => {/*让动物叫*/ });
Action<Dog> actionDog = actionAnimal;
actionDog(aDog);
很明显actionAnimal 是让动物叫,因为Dog是Animal,那么既然Animal 都能叫,Dog肯定也能叫。
In 关键字:逆变,代表输入,代表着只能被使用,不能作为返回值,所以C#编译器可以根据in关键字推断这个泛型类型只能被使用,所以Action<Dog> actionDog = actionAnimal;可以通过编译器的检查。
再次演示Out关键字:
添加两个类:
public interface IMyList<out T>
{
T GetElement();
}
public class MyList<T> : IMyList<T>
{
public T GetElement()
{
return default(T);
}
}
因为out 关键字,所以下面的代码可以通过编译
IMyList<Dog> myDogs = new MyList<Dog>();
IMyList<Animal> myAnimals = myDogs;
将上面的两个类修改为:
public interface IMyList<out T>
{
T GetElement();
void ChangeT(T t);
}
public class MyList<T> : IMyList<T>
{
public T GetElement()
{
return default(T);
}
public void ChangeT(T t)
{
//Change T
}
}
编译:
因为T被out修饰,所以T只能作为参数。
同样修改两个类如下:
public interface IMyList<in T>
{
T GetElement();
void ChangeT(T t);
}
public class MyList<T> : IMyList<T>
{
public T GetElement()
{
return default(T);
}
public void ChangeT(T t)
{
//Change T
}
}
这一次使用in关键字。
编译:
因为用in关键字标记,所以T只能被使用,不能作为返回值。
最后修改代码为:
public interface IMyList<in T>
{
void ChangeT(T t);
}
public class MyList<T> : IMyList<T>
{
public void ChangeT(T t)
{
//Change T
}
}
编译成功,因为in代表了逆变,所以
IMyList<Animal> myAnimals = new MyList<Animal>();
IMyList<Dog> myDogs = myAnimals;
分享到:
相关推荐
1. **接口的协变和逆变**:C# 7.0引入了对接口的协变和逆变支持,例如`IEnumerable<T>`和`IAsyncEnumerable<T>`接口是协变的,这意味着你可以将更具体的类型赋值给更通用的类型。而`IComparer<T>`接口则是逆变的,...
在"GenericPractice"这个文件夹中,可能包含了练习和示例代码,帮助你深入理解并实践C#中的协变和逆变。通过分析和运行这些代码,你将更好地掌握这些高级泛型特性,并能够在实际项目中有效利用它们。
在编程领域,协变和逆变是面向对象编程中的重要概念,特别是在泛型和类型系统的设计中起到...通过阅读“协变和逆变 京华志.txt”以及参考“源码 京华志.url”中的示例,你将能够深入理解这两个概念在实际项目中的应用。
总的来说,理解和熟练运用C#中的协变和逆变对于编写灵活、高效且易于维护的代码至关重要。它们可以帮助你在设计和实现泛型接口、委托和数组时,更好地利用多态性,并提高代码的可扩展性和兼容性。通过深入学习和实践...
- 协变与逆变:这两个概念是理解泛型的关键,本书详细解释了它们的区别以及在实际应用中的意义。 - 匿名方法:与Lambda表达式类似,但更适用于复杂的操作。 4. **案例研究与实践**: - 实战案例:通过具体的例子...
书中详细讲解了泛型的基本用法和高级特性,如约束、协变和逆变等。 4. **委托与事件**:C#中的委托是类型安全的函数指针,事件则是基于委托实现的异步通信机制。这两部分的内容对于理解和使用C#进行事件驱动编程至...
《深入理解C# IN DEPTH (2ND)》是一本针对C#编程语言的高级教程,旨在帮助程序员更深入地了解C#的工作原理。本书由Jon Skeet撰写,覆盖了C# 4版本的所有新特性和改进之处。 #### 二、目标读者与适用场景 - **目标...
在本书中,作者详细讨论了泛型类、泛型方法和泛型接口,同时引入协变和逆变的概念,为程序设计提供了更大的灵活性。 LINQ(语言集成查询)是C#中处理数据的非常强大的工具,它可以让开发者用统一的方式查询和操作...
- **深入理解**:通过对类型系统的深入理解,读者能够更好地设计类和接口,编写出更加健壮的代码。 8. **异常处理(Exception Handling)** - 异常处理是程序健壮性的重要保障。本书详细讲解了C#中的异常处理机制,...
3. 协变与逆变:协变和逆变是泛型编程中的概念,它们允许对泛型类型参数进行更灵活的使用,这在C# 4.0中得到了增强。 4. Lambda表达式:Lambda表达式是C#中实现函数式编程的一个重要特性,它们提供了一种简洁的定义...
- **第十二章:协变和逆变**:这一章解释了协变和逆变的概念及其在泛型编程中的应用。 #### 书籍亮点与收获 - **实用性**:书中提供了大量实用的代码示例,帮助读者将理论知识应用于实际开发过程中。 - **深度...
19. 协变和逆变:在泛型编程中,协变和逆变的概念允许泛型接口和委托在派生类型方面具有一定的灵活性。 20. 高级字符串操作:包括LINQtoObjects的使用,以及如何结合正则表达式进行复杂的字符串处理。 这份C#参考...
5. **泛型协变和逆变**:在C# 4.0中,泛型接口可以声明为协变或逆变,允许在泛型集合间更自由地进行赋值和方法签名的匹配,特别是在处理多态数据时。 6. **匿名类型**:C# 4.0增强了匿名类型的功能,可以用于临时...
C#的泛型支持协变和逆变,这意味着在某些情况下,派生类型的对象可以赋值给基类型引用,即使泛型类型本身不兼容。此外,C#还提供了泛型约束,如where关键字,允许开发者指定类型参数必须遵循的规则,如必须包含某个...
4. **协变和逆变(Covariance and Contravariance)**:C# 4.0支持接口和委托的协变和逆变,这使得类型转换更加灵活,特别是对于泛型接口和委托,能够提高代码的复用性。 5. **多语言互操作(Multilingual ...
泛型的增强,支持协变和逆变,提高了代码的复用性;以及动态类型,允许在运行时确定类型的灵活性。此外,还有Lambda表达式、匿名类型、LINQ(Language Integrated Query)等,极大地提高了开发者的生产力。 5.0中文...
总之,C#语言规范4.0为开发者提供了更强大、更灵活的工具集,包括动态类型、命名和可选参数、泛型的协变和逆变、异步编程支持以及优化的错误处理。对于希望深入理解和掌握C#的进阶学习者来说,仔细研究这份规范文档...
9. **泛型协变和逆变**:C# 3.0对泛型接口和委托增加了协变和逆变的支持,使得类型转换更加灵活,提高了代码的可重用性。 10. **动态类型**:虽然不是C# 3.0的特性,但后续版本C# 4.0引入的动态类型在C# 3.0的上下...