首页后端开发ASP.NETC#中关于逆变和协变的详解

C#中关于逆变和协变的详解

时间2024-01-30 16:51:03发布访客分类ASP.NET浏览247
导读:收集整理的这篇文章主要介绍了C#中关于逆变和协变的详解,觉得挺不错的,现在分享给大家,也给大家做个参考。这篇文章主要为大家详细介绍了C#逆变与协变的相关资料,具有一定的参考价值,感兴趣的小伙伴们可以参考一下该文章中使用了较多的 委托dele...
收集整理的这篇文章主要介绍了C#中关于逆变和协变的详解,觉得挺不错的,现在分享给大家,也给大家做个参考。这篇文章主要为大家详细介绍了C#逆变与协变的相关资料,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

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

在C#从诞生到发展壮大的过程中,新知识点不断引入。逆变与协变并不是C#独创的,属于后续引入。在Java中同样存在逆变与协变,后续我还会写一篇Java逆变协变的文章,有兴趣的朋友可以关注一下。

逆变与协变,听起来很抽象、高深,其实很简单。看下面的代码:


class PErson {
 }
 class Student : Person {
 }
 class Teacher: Person {
 }
  class PRogram {
  static void Main(string[] args)  {
       ListPerson>
     plist = new ListPerson>
    ();
       plist = new ListStudent>
    ();
       plist = new ListTeacher>
    ();
}
}
    

在上面的代码中,plist = new ListStudent> ()、plist = new ListTeacher> ()两句产生编译错误。虽然Person是Student/Teacher的父类,但ListPerson> 类型却不是ListStudent/Teacher> 类型的父类,所以上面的赋值语句报类型转换失败错误。

如上这样的赋值操作,在C# 4.0之前是不允许的,至于为什么不允许,类型安全是首要因素。看下面的示例代码:


ListPerson>
     plist = new ListStudent>
    ();
    plist.Add(new Person());
    plist.Add(new Student());
    plist.Add(new Teacher());
    

如下示例,假设 ListPerson> plist = new ListStudent> () 允许赋值,那plist虽然类型为ListPerson> 集合,但实际指向确是ListStudent> 集合。plist.Add(new Person()),添加操作实际调用的是ListStudent> .Add()。Person类型无法安全转换为Student,所以这样的集合定义没有意义,所以上面的假设不成立。

但情况在C# 4.0之后发生了变化,并不是"不可能发生的事情发生了",而是应用的灵活性做出了新的调整。同样的在C# 4.0中上面的程序仍是不被允许的,但却出现了例外。从C# 4.0开始,在泛型委托、泛型接口中,允许特殊情况的发生(实质上并未发生特殊变化,后面说明)。如下示例:


delegate void WorkT>
    (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)  {
       WorkPerson>
     worker = (p) =>
 {
     Console.WriteLine(p.Name);
 }
    ;
     ;
       WorkStudent>
     student_worker = (s) =>
 {
     Console.WriteLine(s.Like);
 }
    ;
       student_worker = worker;
 //此处编译错误  }
}
    

根据前面的理论支持,student_worker = worker; 的错误很容易理解。但此处我们程序的目的是让 woker 充当 WorkStudent> 的功能,以后调用 student_worker(s)实际调用的是woker(s)。为了满足我们的需求,需要程序做2方面的处理:

1、因在调用student_worker(s)时,实质执行的是woker(s),所以需要s变量的类型能成功转换为woker需要的参数类型。

2、需要告诉编译器,此处允许将 WorkPerson> 类型的对象赋值给 WorkStudent> 类型的变量。

条件1在调用时student_worker(),时编译器会提示要求参数必须是Student类型对象,该对象可成功转换为Person类型对象。

条件2则需要对Woke委托定义进行调整,调整如下:


delegate void WorkInin T>
    (T item);
    

委托名字改为WorkIn是为却别修改前后的委托,关键之处为in T> 。通过增加 in 关键字,标注该泛型委托的类型参数T,仅作为委托方法的参数来使用。此时上面的程序便可成功编译并执行。


delegate void WorkInin T>
    (T item);
class Program {
  static void Main(string[] args)  {
       WorkInPerson>
     woker = (p) =>
 {
     Console.WriteLine(p.Name);
 }
    ;
       WorkInStudent>
     student_worker = woker;
   student_worker(new Student() {
 Name="tom", Like="C#" }
    );
  }
 }
    

对于要求类型参数为子类型,允许赋值类型参数为父类型值的这种情况,称为逆变。逆变在C#中需要用 in 标注泛型的类型参数。逆变虽叫逆变,但只是形式上看似父类对象赋值给子类变量,实质上是方法调用时参数的类型转换。Student s = new Person(),这是不可能的,这不是逆变是错误。

上面的代码如你能转换为下面的形式,那你就可以忘却逆变,本质比现象更重要

以上就是C#中关于逆变和协变的详解的详细内容,更多请关注其它相关文章!

声明:本文内容由网友自发贡献,本站不承担相应法律责任。对本内容有异议或投诉,请联系2913721942#qq.com核实处理,我们将尽快回复您,谢谢合作!

.netcsharp

若转载请注明出处: C#中关于逆变和协变的详解
本文地址: https://pptw.com/jishu/592783.html
浅谈怎么利用node提升工作效率 C#中委托和匿名委托的具体介绍

游客 回复需填写必要信息