服务器之家:专注于VPS、云服务器配置技术及软件下载分享
分类导航

PHP教程|ASP.NET教程|Java教程|ASP教程|编程技术|正则表达式|C/C++|IOS|C#|Swift|Android|VB|R语言|JavaScript|易语言|vb.net|

服务器之家 - 编程语言 - C# - 详解c# 协变和逆变

详解c# 协变和逆变

2022-10-18 11:54gt1987 C#

这篇文章主要介绍了c# 协变和逆变的相关资料,帮助大家更好的理解和学习c#,感兴趣的朋友可以了解下

基本概念

协变:能够使用比原始指定的派生类型的派生程度更大(更具体)的类型。例如 IFoo<父类> = IFoo<子类>
逆变:能够使用比原始指定的派生类型的派生程度更新(更抽象)的类型。例如 IBar<子类> = IBar<父类>

关键字out和in

协变和逆变在泛型参数中的表现方式,out关键字表示协变,in关键字表示逆变。二者只能在泛型接口或者委托中使用。

理解协变和逆变

看完上面的定义是不是一脸懵逼~~~。看不懂就对了,且定义语句的歧义性很大。让我们大脑赶紧清空下!!首先记住一点明确的概念,类的多态展示一定是通过基类来表示,派生的具体类都是可转化为基类,而不能走相反的流程。
下面我们用代码直观的表现下协变和逆变。

?
1
2
3
4
5
6
7
8
9
10
11
12
public class Animal
{
  public void Eat()
  { }
}
 
public class Dog : Animal
{
  public void Run()
  {
  }
}

这是一段很简单的子类和父类的关系,我们进行一下简单的转化,应该很好理解,Dog子类可以用Animal父类展示,反过来则不可以,会编译错误。

?
1
2
3
4
5
Dog dog = new Dog();
Animal animal = dog;
 
//error 编译错误
//Dog dog2 = animal;

那么我们做一点变化。

?
1
2
3
4
5
6
List<Dog> dogs = new List<Dog>();
//error 编译错误
//List<Animal> animals_2 = dogs;
 
IEnumerable<Dog> dogs_2 = dogs;
IEnumerable<Animal> animals = dogs_2;

感觉到一点问题没?Dog子类可以用Animal父类展示,使用List泛型就不可以了,但是IEnumerable泛型又可以。List<>和IEnumerable<>有什么不同?我们看下二者的定义即可发现端倪。

?
1
2
3
4
5
6
7
//IList定义
public interface IList<[NullableAttribute(2)] T> : ICollection<T>, IEnumerable<T>, IEnumerable
{}
 
//和IEnumerable定义
public interface IEnumerable<[NullableAttribute(2)] out T> : IEnumerable
{}

区别就在于 IEnumerable的泛型参数用了out协变标注,所以可以做正确的转换。 这里也可以理解出什么时候需要使用in、out关键字:当你设计带有泛型的基类且泛型类型可能存在扩展时,则需要考虑使用in或者out关键字修饰。
我们再看看官方的Action<>和Func<>类对协变和逆变的使用,先看定义:

?
1
2
3
public delegate void Action<[NullableAttribute(2)] in T>(T obj);
 
public delegate TResult Func<[NullableAttribute(2)] in T, [NullableAttribute(2)] out TResult>(T arg);

Action的泛型类型是入参,用in表示逆变,Func的第二个泛型类型TResult是出参,用out表示协变。
那么这样看起来对in、out关键字的认识就很简单明了了。看看转换示例:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
Action<Dog> action_dog = d => d.Run();
Action<Animal> action_animal = a => a.Eat();
 
//error 编译错误。in
//Action<Animal> action_animal_2 = action_dog;
    //Action泛型多态化
Action<Dog> action_dog_2 = action_animal;
 
Func<int, Dog> func_dog = a => { return new Dog(); };
Func<int, Animal> func_animal = a => { return new Animal(); };
 
    //Func泛型多态化
Func<int, Animal> func_animal_2 = func_dog;
//error 编译错误。out
//Func<int, Dog> func_dog_2 = func_animal;

注意注释编译错误的语句,符合上面我们转换的规则。对于入参,扩展类可以替代基类参数输入,用in修饰;对于出参,扩展类可以替代基类返回输出,用out修饰。相反则都不可以。

最后简单总结下:

  • 什么是协变/逆变?不要去想官方定义!!!!只要记住out是协变,in是逆变即可。
  • 为什么需要使用协变-out、逆变-in。在泛型或委托中,如果不使用协变/逆变,那么泛型类型一个精确的、固定的某一类型。而使用协变/逆变的话,则泛型类型可以实现多态化。但必须区分入参使用in,出参使用out。

以上就是详解c# 协变和逆变的详细内容,更多关于c# 协变和逆变的资料请关注服务器之家其它相关文章!

原文链接:https://www.cnblogs.com/gt1987/p/13143975.html

延伸 · 阅读

精彩推荐
  • C#C#实现窗体间传值实例分析

    C#实现窗体间传值实例分析

    这篇文章主要介绍了C#实现窗体间传值的方法,结合实例形式较为详细的分析了C#针对窗体间传值的处理技巧,具有一定参考借鉴价值,需要的朋友可以参考下...

    Jimmy.Yang7482021-11-04
  • C#C#警惕匿名方法造成的变量共享实例分析

    C#警惕匿名方法造成的变量共享实例分析

    这篇文章主要介绍了C#警惕匿名方法造成的变量共享,以实例形式分析了C#的匿名方法造成变量共享的原因及对应的解决方法,具有一定参考借鉴价值,需要的朋...

    老赵11162021-11-02
  • C#C# 启动 SQL Server 服务的实例

    C# 启动 SQL Server 服务的实例

    下面小编就为大家分享一篇C# 启动 SQL Server 服务的实例,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧...

    yuchen_love6112022-02-13
  • C#C#中SQL Command的基本用法

    C#中SQL Command的基本用法

    今天小编就为大家分享一篇关于C#中SQL Command的基本用法,小编觉得内容挺不错的,现在分享给大家,具有很好的参考价值,需要的朋友一起跟随小编来看看...

    Czhenya7732022-03-03
  • C#为IObservable实现自己的运算符(详解)

    为IObservable实现自己的运算符(详解)

    下面小编就为大家带来一篇为IObservable实现自己的运算符(详解)。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧...

    C#教程网8252022-01-04
  • C#C#正则表达式的6个简单例子

    C#正则表达式的6个简单例子

    本文介绍了C#中的正则表达式的六个例子,都是经常用到的,希望通过本文的介绍,能够给你带来收获。...

    C#教程网6642021-11-01
  • C#C#实现类型的比较示例详解

    C#实现类型的比较示例详解

    这篇文章主要给大家介绍了关于C#实现类型的比较的相关资料,文中通过示例代码介绍的非常详细,对大家学习或者使用C#具有一定的参考学习价值,需要的...

    solenovex6072022-07-21
  • C#C#实现实体类和XML相互转换

    C#实现实体类和XML相互转换

    这篇文章主要为大家详细介绍了C#实现实体类和XML相互转换的资料,具有一定的参考价值,感兴趣的小伙伴们可以参考一下...

    .NET开发菜鸟7912021-12-28