返回顶部
首页 > 资讯 > 精选 >C#中的协变与逆变接口怎么实现
  • 692
分享到

C#中的协变与逆变接口怎么实现

2023-07-05 06:07:05 692人浏览 薄情痞子
摘要

今天小编给大家分享一下C#中的协变与逆变接口怎么实现的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。协变协变概念令人费解,多半

今天小编给大家分享一下C#中的协变与逆变接口怎么实现的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。

协变

协变概念令人费解,多半是取名或者翻译的锅,其实是很容易理解的。

比如大街上有一只狗,我说大家快看,这有一只动物!这个非常自然,虽然动物并不严格等于狗,但不会有人觉得我说的不对,把狗变成动物就是协变,C#也支持这个:

// C#6顶级语句Dog dog= new Dog();Animal animal= dog;interface Animal{}class Dog : Animal{}

那么接下来,大街上有一群狗,我说有一群动物,按理说也是对的,但看样子C#不这么认为

List<Dog> dogLst = new List<Dog>();List<Animal> aniLst = dogLst;       //飙红飙红飙红了interface Animal {}class Dog : Animal {}

原因其实很容易理解,毕竟在上述的代码中,写了Dog:Animal,即声明了狗是动物的子类,但是并没有写List<Animal> : List<Dog>,换言之,从来没有声明过一群狗是一群动物的子类。

但是,如果不用List,而用其父类IEnumerable,写成下面这样,就又不报错了。

List<Dog> dogLst = new List<Dog>();IEnumerable<Animal> aniLst = dogLst;

换言之,C#承认List<Dog>是IEnumerable<Animal>的子类,个中差别,只需一览源码,就会知晓:

public interface IEnumerable<out T> : IEnumerablepublic class List<T> : ..., IEnumerable<T>, ...

IEnumerable无非比List多了一个out参数,有了这个参数,就拥有了协变的功能,从而当U是T的子类时,可以支持IEnumerable<U>到IEnumerable<T>的转换。

在官方文档中,指明了具有out关键字的泛型接口包括IEnumerable<T>, IEnumerator<T>, IQueryable<T>和IGrouping<TKEy,TElement>。

协变接口的实现

协变和逆变目前只能在泛型接口和委托中使用,下面新建一个泛型接口,并使用关键字out。由于使用.net6.0的顶级语句,所以接口和类的声明放在后面。

iout<string> outStr = new Out();IOut<object> outObj = outStr;Console.WriteLine(outObj.getName());interface IOut<out T>{    T getName();}class Out : IOut<string>{    public string getName()    {        return GetType().Name;    }}

编译运行,最后输出Out,即outObj尽管在声明的时候用的是IOut<object>,但在IOut的out修饰符的作用下,成功让IOut<object>变成了IOut<string>的父类,得以顺利调用Out中的方法。

那么接下来,如果想让getName更加完备一些,例如要求实现getName(T name)这样的功能,那么经out修饰的协变接口就无能为力了,像下面这样的写法果然被无情地飙红了

interface IOut<out T>{    void getName(T name);}

逆变

VS作为宇宙顶级IDE,协变逆变十分拎得清,上述代码在飙红的同时,直接给出如下错误

变型无效: 类型参数“T”必须是在“IOut.getName(T)”上有效的 逆变式。“T”为 协变。

换言之,如果想让泛型接口可以输入泛型参数,那么需要用到逆变,具体写法如下,其中修饰符in表示逆变

IIn<object> inObj = new In();IIn<string> inStr = inObj;inStr.getName("in");interface IIn<in T>{    void getName(T name);}class In : IIn<object>{    public void getName(object name)    {        Console.WriteLine(name);    }}

逆变和协变最大的不同,并非in和out这两个修饰符的字数,而是整个替换逻辑发生了变化,上述代码中,实际上是作为子类的string调用了通过父类object作为参数定义的函数。

里氏替换原则

在具体实现了协变与逆变之后,总觉得那里怪怪的,最怪的其实还是下面这行代码的错误

//错错错错错错错错错错错错错错错错错错错错错错错错错错interface IOut<out T>{    void getName(T name);}

而且可以想象,与之相对应的下面的逆变代码也是不对的

//错错错错错错错错错错错错错错错错错错错错错错错错错错interface IOut<in T>{    T getName();}

接下来复盘一下产生这种现象的原因,为了破除命名带来的困扰,接下来考虑泛型接口I<T>,其中有一个函数T test(T t)。现有两个特定的继承自泛型接口I<A>和I<B>的类,假设I<A>要调用I<B>中的方法,那么其流程如下

A I<A>.test(A t),即输入一个A类型的参数

将这个A类型的参数t,传入到B I<B>.test(B t)。由于I<B>要求输入B类型的参数,所以要求A可以转换为B类型。

B I<B>.test(B t)计算完毕,返回一个B类型的参数

这个B类型的参数又被返回给最初的调用者A I<A>.test,而这时I<A>的函数最终将返回一个A类型的参数,换言之,在这个步骤,要求B可以转换为A。

A能转为B,然后还得B能转为A,同时A和B还不相等,这显然是不可能的。

所以逆变和协变分别实现了第2步和第4步。

如果I<A>想要调用I<B>test(B t)中的函数,那么A类型必须可以转成B类型。正如string可以转为object一样,此即逆变,用in修饰,其作用场合为子类调用父类中的方法。

如果I<A>想要调用B I<B>test(),那么作为返回值的B类型必须可以转化为A类型,此即协变,用out修饰,正是父类调用子类的方法。

协变和逆变的统一之处在于,二者都严格遵循这子类可以转变为父类的规则,此即里氏替换。这是1987年,芭芭拉&middot;利斯科夫提出的,她也是2008年图灵奖得主。

在协变逆变的过程中,对里氏替换的遵循主要表现在当子类方法重载父类方法时

  • 方法的输入参数要更加宽松,此即逆变(IOut<object>调用IOut<string>,object比string更宽松)

  • 方法的返回值要更加严格,此即协变(IOut<string>调用IOut<object>,string比object更严格)

以上就是“C#中的协变与逆变接口怎么实现”这篇文章的所有内容,感谢各位的阅读!相信大家阅读完这篇文章都有很大的收获,小编每天都会为大家更新不同的知识,如果还想学习更多的知识,请关注编程网精选频道。

--结束END--

本文标题: C#中的协变与逆变接口怎么实现

本文链接: https://lsjlt.com/news/350054.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

猜你喜欢
  • C#中的协变与逆变接口怎么实现
    今天小编给大家分享一下C#中的协变与逆变接口怎么实现的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。协变协变概念令人费解,多半...
    99+
    2023-07-05
  • C#中的协变与逆变怎么实现
    本篇内容介绍了“C#中的协变与逆变怎么实现”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!前言在C#编程中,由于存在类型之间的强制转换,很容易...
    99+
    2023-06-29
  • C#泛型接口的协变和逆变怎么实现
    本文小编为大家详细介绍“C#泛型接口的协变和逆变怎么实现”,内容详细,步骤清晰,细节处理妥当,希望这篇“C#泛型接口的协变和逆变怎么实现”文章能帮助大家解决疑惑,下面跟着小编的思路慢慢深入,一起来学习新知识吧。1、什么是协变、逆变?假设:T...
    99+
    2023-06-29
  • C#泛型接口的协变和逆变
    1、什么是协变、逆变? 假设:TSub是TParent的子类。协变:如果一个泛型接口IFoo<T>,IFoo<TSub>可以转换为IFoo<TParen...
    99+
    2024-04-02
  • C#中的协变与逆变小结
    一:什么是协变与逆变 协变指能够使用比原始指定的派生类型的派生程度更大(更具体的)的类型,逆变指能够使用比原始指定的派生类型的派生程度更小(不太具体的)的类型 只有泛型接口和泛型委托...
    99+
    2024-04-02
  • C#实现协变和逆变案例
    关于协变逆变,SolidMango的解释是比较可取的。有了协变,比如,在需要返回IEnumerable<object>类型的时候,可以使用IEnmerable<st...
    99+
    2022-11-13
    C# 协变 逆变
  • 图文详解C#中的协变与逆变
    目录前言协变和逆变总结前言 这篇文章简单说说C#中的协变和逆变。 在C#编程中,由于存在类型之间的强制转换,很容易会出现所谓的类型可变性说法,存在协变、逆变、不变三种。 就比如前一篇...
    99+
    2024-04-02
  • C#中协变与逆变的示例分析
    这篇文章主要介绍了C#中协变与逆变的示例分析,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。一:什么是协变与逆变协变指能够使用比原始指定的派生类型的派生程度更大(更具体的)的类...
    99+
    2023-06-25
  • 怎么理解Java中的逆变与协变
    这篇文章主要介绍“怎么理解Java中的逆变与协变”,在日常操作中,相信很多人在怎么理解Java中的逆变与协变问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”怎么理解Java中的逆变与协变”的疑惑有所帮助!接下来...
    99+
    2023-06-02
  • 一文带你了解C#中的协变与逆变
    目录协变协变接口的实现逆变里氏替换原则协变 协变概念令人费解,多半是取名或者翻译的锅,其实是很容易理解的。 比如大街上有一只狗,我说大家快看,这有一只动物!这个非常自然,虽然动物并不...
    99+
    2023-02-26
    C# 协变 逆变 C# 协变 C# 逆变
  • C#泛型的逆变协变是什么
    这篇文章主要介绍“C#泛型的逆变协变是什么”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“C#泛型的逆变协变是什么”文章能帮助大家解决问题。一般来说, 泛型的作用就类似一个占位符, 或者说是一个参数,...
    99+
    2023-07-05
  • 浅谈Java中的桥接方法与泛型的逆变和协变
    目录1. 泛型的协变1.1 泛型协变的使用1.2 泛型协变存在的问题1.2.1 Java当中桥接方法的来由1.2.2 为什么泛型协变时,不允许添加元素呢1.2.3 从Java字节码的...
    99+
    2024-04-02
  • C#中逆变的实际应用场景详解
    目录前言协变的应用场景逆变的应用场景讨论总结前言 早期在学习泛型的协变与逆变时,网上的文章讲解、例子算是能看懂,但关于逆变的具体应用场景这方面的知识,我并没有深刻的认识。本文将在具体...
    99+
    2024-04-02
  • C#指针变量与unsafe的实现
    目录指针变量使用指针检索数据的值将指针作为参数传递给函数使用指针访问数组元素编译不安全代码为了保持类型的安全性,默认情况下 C# 是不支持指针的,但是如果使用 unsafe 关键字来...
    99+
    2023-05-14
    C#指针变量 C# unsafe
  • C#中怎么实现多个接口
    今天就跟大家聊聊有关C#中怎么实现多个接口,可能很多人都不太了解,为了让大家更加了解,小编给大家总结了以下内容,希望大家根据这篇文章可以有所收获。为了C#实现多个接口,我们可以从接口派生类。这样的派生类必须为所有接口的方法提供实现,除非派生...
    99+
    2023-06-17
  • C++中怎么实现一个接口
    C++中怎么实现一个接口,针对这个问题,这篇文章详细介绍了相对应的分析和解答,希望可以帮助更多想解决这个问题的小伙伴找到更简单易行的方法。定理1:C++接口是依赖的终点。接口不需要依赖任何东西。推论1:依赖C++接口是安全的。不会带来更多的...
    99+
    2023-06-17
  • PHP Session 跨域与API接口的协作实现
    随着Web应用程序的迅速发展,不同域名之间的数据交互变得越来越常见。在实现跨域数据交互时,PHP的Session机制和API接口的使用成为了解决问题的有效途径。本文将介绍如何在PHP中实现Session跨域以及如何与API接口协作,同时提供...
    99+
    2023-10-21
    API PHP session
  • C语言怎么实现radon变换
    C语言实现Radon变换的步骤如下: 首先,你需要定义一个输入图像的二维数组,并初始化图像的像素值。 创建一个与输入图像等大...
    99+
    2023-10-23
    C语言
  • C#怎么实现InterfaceImplementer.cs接口
    这篇“C#怎么实现InterfaceImplementer.cs接口”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“C#怎么实...
    99+
    2023-06-17
  • C#的Websocket怎么连接实现wss协议
    本篇内容主要讲解“C#的Websocket怎么连接实现wss协议”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“C#的Websocket怎么连接实现wss协议”吧!一、什么是Websocket?W...
    99+
    2023-06-30
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作