C#泛型的逆变协变(个人理解)

前编

一般来说, 泛型的作用就类似一个占位符, 或者说是一个参数, 可以让我们把类型像参数一样进行传递, 尽可能地复用代码

我有个朋友, 在使用的过程中发现一个问题

IFace<object> item = new Face<string>(); // CS0266

public interface IFace<T>
{
    string Print(T input);
}
public class Face<T> : IFace<T>
{
    public string Print(T input) => input.ToString();
}

Q:   string 明明是 object 的子类, 为啥这样赋值会报错呢???
A:   因为 Face<string> 实现的是 IFace<string>, 而 IFace<string> 并不是 IFace<object> 的子类
Q:   但是 stringobject 的子类啊, IFace<string> 可不就是 IFace<object> 吗?
A:   如果只论接口定义, 看起来确实是这样的, 但是你要看内部实现的方法, IFace<string>Print 方法参数是 string, 但是 IFace<object>Print 参数是 object, 如果上面的赋值可以成立, 就意味着允许 Print(string input) 方法传递任意类型的对象, 这样明显是有问题的
Q:   但是我曾经看到过 IEnumerable<object> list = new List<string>(); 这个为什么就可以
A:   这就要讲到C#泛型里的逆变协变
Q:   细嗦细嗦

逆变协变

C#泛型中的逆变(in)协变(out)对于不常自定义泛型的开发来说(可能)是个很难理解的概念, 简单来说其表现形式如下

逆变(in): I<子类> = I<父类>
协变(out): I<父类> = I<子类>

上面例子中提到的 IEnumerable<object> list = new List<string>(); 体现的是协变, 符合一般直觉, 整体上看起来就像是将子类赋值给基类

转到 IEnumerable<> 的定义, 我们可以看到

public interface IEnumerable<out T> : IEnumerable
{
    new IEnumerator<T> GetEnumerator();
}

泛型 T 之前加了协变的关键词 out, 代表支持协变, 可以进行符合直觉且和谐的转化

前编中提到的代码例子不适用并且也不能改造成协变, 只适合使用逆变

相比于符合直觉且和谐协变, 逆变不符合直觉并且别扭

IFace<string> item = new Face<object>();

public interface IFace<in T>
{
    string Print(T input);
}
public class Face<T> : IFace<T>
{
    public string Print(T input) => input.ToString();
}

这是一个逆变的例子, 与协变相似, 需要在泛型 T 之前加上关键词 in

对比上方的协变, 逆变看起来就像是将基类赋值给子类, 但这其实符合里氏代换的

当我们调用 item.Print 时, 看起来允许传入的参数为 string 类型, 而实际上最终调用的 Face<object>.Print 是支持 object 的, 传入 string 类型的参数没有任何问题

逆变协变的作用

逆变(in)协变(out)的作用就是扩展泛型的用法, 帮助开发者更好地复用代码, 同时通过约束限制可能会出现的破坏类型安全的操作

逆变协变的限制

虽然上面讲了逆变(in)协变(out)看起来是什么样的, 但我的那个朋友还是有些疑问

Q:   那我什么时候可以用逆变, 什么时候可以用协变, 这两个东西用起来有什么限制?
A:   简单来说, 有关泛型输入的用逆变, 关键词是in, 有关泛型输出的用协变, 关键词是out, 如果接口中既有输入又有输出, 就不能用逆变协变
Q:   为什么这两个不能同时存在?
A:   协变的表现形式为将子类赋值给基类, 当进行输出相关操作时, 输出的对象类型为基类, 是将子类转为基类, 你可以说子类是基类;
逆变的表现形式为将基类赋值给子类, 当进行输入相关操作时, 输入的对象为子类, 是将子类转为基类, 这个时候你也可以说子类是基类;
如果同时支持逆变协变, 若先进行子类赋值给基类的操作, 此时输出的是基类, 子类转为基类并不会有什么问题, 但进行输入操作时就是在将基类转为子类, 此时是无法保证类型安全的;
Q:   听不懂, 能不能举个例子给我?
A:   假设 IEnumerable<> 同时支持逆变协变, IEnumerable<object> list = new List<string>();进行赋值后, list中实际保存的类型是string, item.First()输出类型为object, 实际类型是string, 此时说stringobject没有任何问题, 协变可以正常发挥作用;
但是如果支持了逆变, 假设我们进行输入类型的操作, item.Add() 允许的参数类型为 object, 可以是任意类型, 但是实际上支持string类型, 此时的object绝无可能是string
Q:   好像听懂了一点了, 我以后慢慢琢磨吧

两者的限制简单总结就是

输入的用逆变
输出的用协变

原文链接:https://www.cnblogs.com/CollapseNav/p/17285595.html

本站文章如无特殊说明,均为本站原创,如若转载,请注明出处:C#泛型的逆变协变(个人理解) - Python技术站

(0)
上一篇 2023年4月18日
下一篇 2023年4月18日

相关文章

  • C#中DataTable和List互转的示例代码

    下面我将详细讲解“C#中DataTable和List互转的示例代码”的完整攻略。 目录 DataTable转List 1.1 使用ToList扩展方法 1.2 使用反射自动映射 List转DataTable 2.1 使用数据表生成方式 2.2 使用反射自动映射 1. DataTable转List 1.1 使用ToList扩展方法 public static …

    C# 2023年5月31日
    00
  • C#调用pyd的方法

    当我们需要使用Python库的时候,可以通过C#代码调用Python库提供的功能。Python库通常是以.so 或 .pyd 的文件形式提供,因此,我们需要使用C#的相关机制调用 Python库。下面将介绍如何在C#中调用Python库的方法。 步骤一: 安装Python 我们需要在计算机上安装Python,并添加Python的安装目录到系统路径中。可以通过…

    C# 2023年6月3日
    00
  • C#线程执行超时处理与并发线程数控制实例

    首先,我们需要明确一下本篇攻略的主要内容,即是如何应对C#程序中的线程执行超时问题以及控制并发线程数。接下来,我们将分几个方面来逐一介绍相关的方法和实例。 线程执行超时处理 在C#多线程编程中,一个常见的问题就是线程运行时间过长导致程序性能下降或死锁。为了解决这个问题,我们可以使用一个超时处理机制,即线程运行时间超过一定时间就强制终止线程,避免出现程序僵死的…

    C# 2023年5月15日
    00
  • c# 死锁和活锁的发生及避免

    C# 死锁和活锁的发生及避免攻略 什么是死锁和活锁 死锁和活锁都是多线程并发编程中经常遇到的问题。 死锁 死锁指的是两个或更多的线程被永久地阻塞,无法继续执行,因为每个线程都在等待其他线程释放资源。简单来说,就是线程之间互相占用对方需要的资源,并不释放,而导致程序无限等待下去。 活锁 活锁指的是线程虽然没有被阻塞,但是他们却无法继续前进,因为它们总是在响应其…

    C# 2023年6月7日
    00
  • C#多线程系列之任务基础(二)

    下面是关于”C#多线程系列之任务基础(二)”的详细讲解。 标题 C#多线程系列之任务基础(二) 代码块 var task = Task.Run(() => { // 这里是异步执行的任务代码 }); 正文 本文主要讲解了在C#中利用Task实现多线程编程的基础知识。在任务基础(一)中,我们讲解了Task的基本概念、使用方法以及几种等待任务完成的方法。在…

    C# 2023年6月3日
    00
  • 效控制C#中label输出文字的长度,自动换行

    效控制C#中label输出文字的长度,自动换行的方法: 使用AutoEllipsis属性 可以使用C#中的Label控件中的AutoEllipsis属性实现标签控件中输出文字的长度的控制。在Winform应用程序中,将AutoEllipsis属性设置为true即可实现标签文字长度过长时的自动省略号替换。示例代码如下: label1.AutoEllipsis …

    C# 2023年6月7日
    00
  • C#封装DBHelper类

    下面是我对“C#封装DBHelper类”的完整攻略: 第一步:创建封装类 首先,我们需要创建一个名为DBHelper的类,这个类将会是一个对应于一个数据库连接的封装,提供了一系列的方法来操作数据库。这个类可以采用单例模式,确保整个应用程序只会有一个数据库连接对象。以下是一个简单的DBHelper类的示例: using System; using System…

    C# 2023年5月31日
    00
  • C#中委托和事件的区别详解

    C#中委托和事件的区别详解 什么是委托和事件 委托 委托(delegate)是一种类型,它可以代表多个方法,并且只有这些方法的签名一致才能被委托代表。委托可以看做是方法的引用,提供了一种将方法作为参数传递给其他方法的方式。 在C#中声明一个委托类型,需要使用delegate关键字。 事件 事件(event)是委托的一种应用,它允许对象在某个事件发生时,通知其…

    C# 2023年6月7日
    00
合作推广
合作推广
分享本页
返回顶部