C#中的协变与逆变小结

作者:Hello Bug. 时间:2022-10-25 13:30:25 

一:什么是协变与逆变

协变指能够使用比原始指定的派生类型的派生程度更大(更具体的)的类型,逆变指能够使用比原始指定的派生类型的派生程度更小(不太具体的)的类型

只有泛型接口和泛型委托参数支持协变和逆变

二:引言


using System;
using System.Collections.Generic;

class MainClass
{
   static void Main()
   {
       object o = "str";
       List<object> oList = new List<string>();
       IEnumerable<object> strs = new List<string>();
   }
}

上面这段代码中,第一句没问题,属于类型安全转换,第二句会报错,因为这两个list并没有继承关系,而第三句是正确的,其实在背后,就是协变和逆变在起作用

三:协变

协变在泛型方法的参数里以out表示,使用out可以在声明父类泛型参数的时候使用子类泛型参数构造,out参数可以单纯的理解为输出,作为返回值例如IEnumerable<T>接口

C#中的协变与逆变小结


using System;
using System.Collections.Generic;

class MainClass
{
   static void Main()
   {
       IEnumerable<object> list = new List<string>();
   }
}

分析一下上面的代码为什么是合法的呢?首先虽然是用IEnumerable<object>声明的,但是是用List<string>构造的,列表中的元素是string类型。其次IEnumerable的作用只有遍历元素,不允许添加操作,所以是合法的,本质上就是里氏替换原则

四:逆变

逆变在泛型方法的参数里以in表示,使用in可以在声明子类泛型参数的时候使用父类泛型参数构造,int参数只能作为传入值不能作为返回值例如Action<T>委托

C#中的协变与逆变小结


using System;

class MainClass
{
   static void Main()
   {
       Action<string> action = new Action<object>((o)=> { });
       action("");
   }
}

分析一下上面的代码为什么是合法的呢?看似是object转换成了string,但实际上使用委托的时候传入的是一个string类型的参数,然后将string转换成了object,本质上还是派生类到基类的转换,所以是类型安全的,本质上就是里氏替换原则

五:为什么协变和逆变是针对泛型接口或泛型委托参数的?

不能针对泛型类?

由上可知,协变和逆变都是定义方法成员的(接口不能定义字段只能定义成员),而方法成员在创建对象时是不涉及到对象内存分配的,所以是类型安全的,而泛型类是模板类,类中可以包含字段, 所以是不安全的


using System;
using System.Collections.Generic;

class MainClass
{
   static void Main()
   {
       object o1 = "";//类型安全
       string s1 = (string)o1;//非类型安全
       IEnumerable<object> o2 = new List<string>();//协变
       Action<string> s2 = new Action<object>((o) => { });//逆变
   }
}

六:自定义协变


using System;
using System.Collections.Generic;

class MainClass
{
   static void Main()
   {
       ICustomCovariant<object> o = new CustomCovariant<string>();
   }
}

public interface ICustomCovariant<out T>
{
   T Get();
}
public class CustomCovariant< T> : ICustomCovariant<T>
{
   public T Get()
   {
       return default(T);
   }
}

七:自定义逆变


using System;
using System.Collections.Generic;

class MainClass
{
   static void Main()
   {
       IContravariant<string> o = new CustomContravariant<object>();
   }
}

public interface IContravariant<in T>
{
   void Get(T t);
}
public class CustomContravariant<T> : IContravariant<T>
{
   public void Get(T t)
   {

}
}

八:总结

——在泛型中,如果确定泛型参数是只读或者只写的,那么就可以使用协变或者逆变。如果泛型参数无法确定只读或只写,这种类型参数既不能协变也不能逆变,只能精确类型匹配 ——在泛型或委托中,如果不使用协变或逆变,那么泛型类型是一个固定类型,而使用协变或逆变的话,则泛型类型可以实现多态化

来源:https://blog.csdn.net/LLLLL__/article/details/120966813

标签:C#,协变,逆变
0
投稿

猜你喜欢

  • Java 集合的Contains和Remove方法

    2022-06-17 05:29:10
  • 详解maven中央仓库连不上的解决办法

    2023-02-25 08:43:54
  • C# WPF ListView控件的实例详解

    2022-08-05 17:54:23
  • Android中banner的使用步骤

    2023-07-04 17:47:23
  • C# 使用multipart form-data方式post数据到服务器

    2023-12-20 21:10:58
  • Java解释器的运行过程介绍

    2022-10-03 01:21:30
  • Java BufferedImage转换为MultipartFile方式

    2022-07-13 10:02:46
  • Spring Boot 集成Shiro的多realm配置过程

    2023-09-17 10:07:26
  • Java 按行读取文件按行写入文件并以空格分割字符串的方法

    2023-08-31 12:32:21
  • 浅析java移位符的具体使用

    2023-12-21 09:36:13
  • 跨域解决方案Jsonp原理解析

    2023-11-07 18:48:05
  • 防止未登录用户操作—基于struts2拦截器的简单实现

    2021-06-11 13:21:00
  • Android列表实现(2)_游标列表案例讲解

    2022-11-15 16:49:55
  • C#中静态方法和实例化方法的区别、使用

    2023-08-25 20:49:33
  • WinForm生成验证码图片的方法

    2022-05-11 17:49:33
  • java内存模型jvm虚拟机简要分析

    2022-09-08 09:29:34
  • C#条件拼接Expression<Func<T, bool>>的使用

    2023-07-11 22:29:18
  • Java中String类的常用方法总结

    2021-11-26 10:39:20
  • 学习Java九大内置对象

    2023-10-13 01:10:10
  • java中Date类和Strng类的灵活转化

    2022-12-09 07:47:24
  • asp之家 软件编程 m.aspxhome.com