该文章中使用了较多的 委托delegate和Lambda表达式,如果你并不熟悉这些,请查看我的文章《委托与匿名委托》、《匿名委托与Lambda表达式》以便帮你建立完整的知识体系。

在C#从诞生到发展壮大的过程中,新知识点不断引入。逆变与协变并不是C#独创的,属于后续引入。在Java中同样存在逆变与协变,后续我还会写一篇Java逆变协变的文章,有兴趣的朋友可以关注一下。
逆变与协变,听起来很抽象、高深,其实很简单。看下面的代码:
class Person
{
}
class Student : Person
{
}
class Teacher: Person
{
}
class Program
{
static void Main(string[] args)
{
List<Person> plist = new List<Person>();
plist = new List<Student>();
plist = new List<Teacher>();
}
}
在上面的代码中,plist = new List<Student>()、plist = new List<Teacher>()两句产生编译错误。虽然Person是Student/Teacher的父类,但List<Person>类型却不是List<Student/Teacher>类型的父类,所以上面的赋值语句报类型转换失败错误。
如上这样的赋值操作,在C# 4.0之前是不允许的,至于为什么不允许,类型安全是首要因素。看下面的示例代码:
List<Person> plist = new List<Student>(); plist.Add(new Person()); plist.Add(new Student()); plist.Add(new Teacher());
如下示例,假设 List<Person> plist = new List<Student>() 允许赋值,那plist虽然类型为List<Person>集合,但实际指向确是List<Student>集合。plist.Add(new Person()),添加操作实际调用的是List<Student>.Add()。Person类型无法安全转换为Student,所以这样的集合定义没有意义,所以上面的假设不成立。
但情况在C# 4.0之后发生了变化,并不是"不可能发生的事情发生了",而是应用的灵活性做出了新的调整。同样的在C# 4.0中上面的程序仍是不被允许的,但却出现了例外。从C# 4.0开始,在泛型委托、泛型接口中,允许特殊情况的发生(实质上并未发生特殊变化,后面说明)。如下示例:
delegate void Work<T>(T item);
class Person
{
public string Name { get; set; }
}
class Student : Person
{
public string Like { get; set; }
}
class Teacher : Person
{
public string Teach { get; set; }
}
class Program
{
static void Main(string[] args)
{
Work<Person> worker = (p) => { Console.WriteLine(p.Name); }; ;
Work<Student> student_worker = (s) => { Console.WriteLine(s.Like); };
student_worker = worker; //此处编译错误
}
}
根据前面的理论支持,student_worker = worker;的错误很容易理解。但此处我们程序的目的是让 woker 充当 Work<Student> 的功能,以后调用 student_worker(s)实际调用的是woker(s)。为了满足我们的需求,需要程序做2方面的处理:
1、因在调用student_worker(s)时,实质执行的是woker(s),所以需要s变量的类型能成功转换为woker需要的参数类型。
2、需要告诉编译器,此处允许将 Work<Person> 类型的对象赋值给 Work<Student>类型的变量。
条件1在调用时student_worker(),时编译器会提示要求参数必须是Student类型对象,该对象可成功转换为Person类型对象。
条件2则需要对Woke委托定义进行调整,调整如下:
delegate void WorkIn<in T>(T item);
委托名字改为WorkIn是为却别修改前后的委托,关键之处为<in T>。通过增加 in 关键字,标注该泛型委托的类型参数T,仅作为委托方法的参数来使用。此时上面的程序便可成功编译并执行。
delegate void WorkIn<in T>(T item);
class Program
{
static void Main(string[] args)
{
WorkIn<Person> woker = (p) => { Console.WriteLine(p.Name); };
WorkIn<Student> student_worker = woker;
student_worker(new Student() { Name="tom", Like="C#" });
}
}
对于要求类型参数为子类型,允许赋值类型参数为父类型值的这种情况,称为逆变。逆变在C#中需要用 in 标注泛型的类型参数。逆变虽叫逆变,但只是形式上看似父类对象赋值给子类变量,实质上是方法调用时参数的类型转换。Student s = new Person(),这是不可能的,这不是逆变是错误。
上面的代码如你能转换为下面的形式,那你就可以忘却逆变,本质比现象更重要😀:
delegate void WorkIn<in T>(T item);
class Program
{
static void Main(string[] args)
{
WorkIn<Person> woker = (p) => { Console.WriteLine(p.Name); };
WorkIn<Student> student_worker = (s)=> { woker(s); };
student_worker(new Student() { Name="tom", Like="C#" });
}
}
协变
现在修改我们的程序需求,要求Work委托执行后返回一个Person对象,如下:
delegate T Work<T>();
class Program
{
static void Main(string[] args)
{
Work<Person> worker = () => { return new Person(); };
Work<Student> student_worker = () => { return new Student(); };
worker = student_worker;
}
}
同上 worker = student_worker 无法通过编译,此时我们的目的为:用 Work<Student> student_woker 的功能替代 Work<Person> 的功能,因为 student_woker 执行后返回一个Student对象,这完全符合 Work<Person> 的要求。
如果要实现上面的目的,程序同样需做2方面的处理:
1、因在调用 worker()时,实质执行的是 student_worker(),所以需要 student_worker() 执行结果能功转换为woker 执行后返回的类型。
2、需要告诉编译器,此处允许将 Work<Student>类型的对象赋值给 Work<Person> 类型的变量。
此时条件1,上述代码已经满足,对于条件2,需要泛型委托Work做如下调整:
delegate T WorkOut<out T>();
委托名字改为WorkOut也为却别修改前后的委托,关键之处为<out T>。通过增加 out 关键字,标注该泛型委托的类型参数T,仅作为委托方法的返回值类型来使用。此时上面的程序便可成功编译并执行。
delegate T WorkOut<out T>();
class Program
{
static void Main(string[] args)
{
WorkOut<Person> worker = () => { return new Person(); };
WorkOut<Student> student_worker = () => { return new Student(); };
worker = student_worker;
Person p = worker();
}
}
对于要求泛型类型参数为父类型,允许赋值类型参数为子类型值的这种情况,称为协变。协变在C#中需要用 out 标注泛型的类型参数。
注意:逆变、协变类型说明的区别。根据引出的定义逆变的形式只可能发生在泛型上(泛型接口、泛型委托),而协变的代码形式就比较多,但并不一定是协变。所以在协变中用红色注明,必须是关于泛型参数的情况才是协变。下面这类情况不属于协变(至少我不认为它们是协变):
Person p = new Student();
上面的示例代码如你能转换为下面的形式,那你也可以忘却协变😀:
delegate T WorkOut<out T>();
class Program
{
static void Main(string[] args)
{
WorkOut<Student> student_worker = () => { return new Student(); };
WorkOut<Person> worker = () => { return student_worker (); };
Person p = worker();
}
}
通过上面的内容可以发现,逆变、协变其实是方法参数、返回值类型的转换与对委托方法的包装而已。抓住其核心,再看各种形式的代码就简单了。
在C# 4.0 中 你可以查看 Action,Func的定义,以便更深入理解逆变、协变。
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。
# C#
# 逆变
# 协变
# 一文带你了解C#中的协变与逆变
# C#实现协变和逆变案例
# 一句话清晰总结C#的协变和逆变
# C#泛型接口的协变和逆变
# 图文详解C#中的协变与逆变
# 详析C#的协变和逆变
# C#中的协变与逆变小结
# 详解c# 协变和逆变
# 你了解C#的协变和逆变吗
# 看完这篇就懂了
# 一篇文章看懂C#中的协变、逆变
# C#中的协变与逆变深入讲解
# c#协变和逆变实例分析
# C#4.0新特性之协变与逆变实例分析
# C#泛型的逆变协变之个人理解
# 的是
# 转换为
# 不可能
# 之处
# 你能
# 便可
# 较多
# 这种情况
# 需要用
# 实质上
# 返回值
# 为父
# 这是
# 发生了
# 如果你
# 我不
# 出了
# 你可以
# 才是
相关文章:
Thinkphp 中 distinct 的用法解析
重庆市网站制作公司,重庆招聘网站哪个好?
如何快速建站并高效导出源代码?
建站之星如何开启自定义404页面避免用户流失?
三星网站视频制作教程下载,三星w23网页如何全屏?
孙琪峥织梦建站教程如何优化数据库安全?
建站之星在线客服如何快速接入解答?
北京制作网站的公司排名,北京三快科技有限公司是做什么?北京三快科技?
北京网站制作费用多少,建立一个公司网站的费用.有哪些部分,分别要多少钱?
广平建站公司哪家专业可靠?如何选择?
如何使用Golang table-driven基准测试_多组数据测量函数效率
Python lxml的etree和ElementTree有什么区别
如何在Windows环境下新建FTP站点并设置权限?
建站之星微信建站一键生成小程序+多端营销系统
建站之星下载版如何获取与安装?
XML的“混合内容”是什么 怎么用DTD或XSD定义
小程序网站制作需要准备什么资料,如何制作小程序?
香港服务器网站搭建教程-电商部署、配置优化与安全稳定指南
大学网站设计制作软件有哪些,如何将网站制作成自己app?
如何获取上海专业网站定制建站电话?
网页制作模板网站推荐,网页设计海报之类的素材哪里好?
网站制作说明怎么写,简述网页设计的流程并说明原因?
建站主机选哪家性价比最高?
建站主机空间推荐 高性价比配置与快速部署方案解析
制作网站建设的公司有哪些,网站建设比较好的公司都有哪些?
广州美橙建站如何快速搭建多端合一网站?
长沙企业网站制作哪家好,长沙水业集团官方网站?
制作证书网站有哪些,全国城建培训中心证书查询官网?
宠物网站制作html代码,有没有专门介绍宠物如何养的网站啊?
php能控制zigbee模块吗_php通过串口与cc2530 zigbee通信【介绍】
定制建站流程步骤详解:一站式方案设计与开发指南
网站插件制作软件免费下载,网页视频怎么下到本地插件?
如何在Windows 2008云服务器安全搭建网站?
邀请函制作网站有哪些,有没有做年会邀请函的网站啊?在线制作,模板很多的那种?
ppt在线制作免费网站推荐,有什么下载免费的ppt模板网站?
,网页ppt怎么弄成自己的ppt?
个人摄影网站制作流程,摄影爱好者都去什么网站?
如何在IIS中新建站点并配置端口与物理路径?
盐城做公司网站,江苏电子版退休证办理流程?
如何通过远程VPS快速搭建个人网站?
广州网站制作的公司,现在专门做网站的公司有没有哪几家是比较好的,性价比高,模板也多的?
番禺网站制作公司哪家值得合作,番禺图书馆新馆开放了吗?
已有域名和空间如何快速搭建网站?
小说建站VPS选用指南:性能对比、配置优化与建站方案解析
微信小程序 input输入框控件详解及实例(多种示例)
如何在Windows服务器上快速搭建网站?
如何续费美橙建站之星域名及服务?
广州网站建站公司选择指南:建站流程与SEO优化关键词解析
网站制作价目表怎么做,珍爱网婚介费用多少?
宝塔面板如何快速创建新站点?
*请认真填写需求信息,我们会在24小时内与您取得联系。