首页
学习
活动
专区
工具
TVP
发布
精选内容/技术社群/优惠产品,尽在小程序
立即前往

打字中协变与逆变位置的区别

协变(Covariance)与逆变(Contravariance)是C#泛型中处理类型转换的两个重要概念,它们允许在保持类型安全的前提下,实现泛型类型之间的灵活转换。具体来说:

协变(Covariance)

  • 基础概念:允许使用比原始指定的派生类型的派生程度更大的类型。
  • 关键字:使用out关键字在泛型接口或委托的类型参数上指定。
  • 应用场景:泛型接口或委托的返回类型,如IEnumerable<T>,其中T使用out关键字,表示T只能作为返回值。
  • 示例代码
代码语言:txt
复制
public interface ICovariant<out T> {
    T Get();
}

public class Base { }

public class Derived : Base { }

public class BaseProvider : ICovariant<Base> {
    public Base Get() => new Base();
}

ICovariant<Base> baseProvider = new BaseProvider(); // 正常赋值
ICovariant<Derived> derivedProvider = baseProvider; // 这是允许的,因为BaseProvider实现了ICovariant<Base>

逆变(Contravariance)

  • 基础概念:允许使用比原始指定的派生类型的派生程度更小的类型。
  • 关键字:使用in关键字在泛型接口或委托的类型参数上指定。
  • 应用场景:泛型接口或委托的参数类型,如Action<T>,其中T使用in关键字,表示T只能作为参数。
  • 示例代码
代码语言:txt
复制
public interface IContravariant<in T> {
    void Set(T value);
}

public class Base { }

public class Derived : Base { }

public class BaseConsumer : IContravariant<Base> {
    public void Set(Base value) { /* ... */ }
}

IContravariant<Base> baseConsumer = new BaseConsumer(); // 正常赋值
IContravariant<Derived> derivedConsumer = baseConsumer; // 这是允许的,因为BaseConsumer的Set方法接受Base类型,而Derived是Base的子类

通过上述解释和示例代码,我们可以看到协变与逆变在C#中的实际应用,它们不仅增加了代码的灵活性,还确保了类型安全。

页面内容是否对你有帮助?
有帮助
没帮助

相关·内容

协变、逆变与不变

如果 T[A] 是 T[B] 的子类,则这种关系是「逆变」,因为参数化类型 T 的父子类关系与类型参数的父子类关系是「相反方向的」。...在 Scala 中在类型参数前添加 + 代表参数化类型在该类型参数上协变,添加 - 则代表逆变,什么都不加就是不变。...在 Java 中,类似于协变,逆变也是在应用的时候才去对其进行约束,例如: List persons = new ArrayList(); List与协变的情况是类似的。 于是,Scala 和 Java 的型变标记可以进行如下总结 3: Scala Java 解释 +T ?...在 Scala 中,如果进行了协变或者逆变的标记,编译器就会对这个类型参数的使用进行检查,如果它出现在了错误的位置上,编译器就会提示错误,防止了开发者因此而犯错。

1.9K30

TypeScript 中的逆变、协变和双向协变

通过协变和逆变原则 协变与逆变(covariance and contravariance)是在计算机科学中,描述具有父/子型别关系的多个型别通过型别构造器、构造出的多个复杂型别之间是否有父/子型别关系的用语...维基百科上关于协变和逆变的解释有点晦涩难懂。...但在TS中,参数类型是双向协变的(详见下文3.1小节),如果项目里开启了"strict": true,意味着,会来带开启 strictFunctionType ,此时,才按照逆变处理 双向协变 在老版本的...TS 中,函数参数是双向协变的。...也就是说,既可以协变又可以逆变,但是这并不是类型安全的。在新版本 TS (2.6+) 中 ,你可以通过开启 strictFunctionTypes 或 strict 来修复这个问题。

1.2K20
  • “协变”、“逆变”与Delegate类型转换

    之后又想到了其他一些相关的东西,除了简单地分析如何通过Emit实现EventHandler的类型转换之外,还加上关于Delegate“协变”与“逆变”的一些东西,算是对前一篇文章的完善。...目录 一、从Delegate的“协变”与“逆变”说起 二、EventHandler是否换一种定义方式更好?...四、通过Emit实现EventHandler的类型转换 五、最简单的转换方式 一、从Delegate的“协变”与“逆变”说起 根据Delegate“协变”与“逆变”的原理,对于两个具有相同声明的两个...我们在定义泛型Delegate的时候可以利用C#“协变”与“逆变”,使类型为A对象能够赋值给类型为B的变量。...如果事件类型对于得Delegate并没有采用逆变方式定义,那么要求我们注册一个与之类型完全一致的Delegate。

    86770

    Java泛型的协变与逆变

    导读 泛型是Java最基础的语法之一,众所周知:出于安全原因,泛型默认不能支持型变(否则会引入危险),因此Java提供了通配符上限和通配符下限来支持型变,其中通配符上限就泛型协变,通配符下限就是泛型逆变...因此对于协变的泛型集合,程序只能从集合中取出元素——取出的元素的类型肯定能保证是上限;但程序不能向集合添加元素——因此程序无法确定程序要求的集合元素具体是上限的哪个子类。...extends Number>是支持协变的泛型,因此程序中两行①号代码可以分别将List、List赋值给List类型的变量。...因此对于逆变的泛型集合,程序只能向集合中添加元素——添加元素的类型总能符合上限——而集合元素总是上限的父类,因此完全没问题;但程序不能从集合中取出元素——因为编译器无法确定集合元素具体是下限的哪个父类—...但如果程序尝试从泛型逆变的集合中取出元素,那么取出的元素只能被当成Object处理(众生皆Object)。

    1.3K40

    Swift 之类型的协变与逆变

    —沃茨•其索特 1 什么是协变与逆变 刚开始看到协变(Covariance)和逆变(Contravariance)的时候,差点晕菜,反复查了一些资料,才稍有些自己的体会,难免有理解不对的地方,欢迎指出...那么我们就可以这么来解释协变和逆变了: 协变: 如果说 List 也是 List的subtype,也就是衍生类型的关系和原来类型( Cat 与 Animal)的关系是一致的,那我们就说...不变:如果说List 既不是 List的subtype,也不是supertype,也就是说没有关系,则说是不变的。 2 为什么要了解协变与逆变?...() -> Cat 和 () -> Animal 的关系与 Cat 和 Animal 之间的关系一致,也就是说是在 Swift 中函数的返回值是协变的。...其他类型的协变和逆变 上面我们提到了函数的参数和返回值的分别是逆变和协变,在 Swift 中除了函数,还有属性(property),范型(Generic)等。

    92620

    C#进阶-协变与逆变

    我们知道子类转换到父类,在C#中是能够隐式转换的。这种子类到父类的转换就是协变。而另外一种类似于父类转向子类的变换,可以简单的理解为逆变。...逆变协变可以用于泛型委托和泛型接口,本篇文章我们将讲解C#里逆变和协变的使用。逆变和协变的语法第一次接触难免感到陌生,最好的学习方式就是在项目中多去使用,相信会有很多感悟。...协变与逆变 协变(共变):泛型委托或泛型接口的类似于父类转向子类的变换; 逆变(反变):泛型委托或泛型接口的类似子类到父类的隐式转换; 逆变与协变用来描述类型转换后的继承关系,其定义:如果A、B表示类型...令List成为List子类的变化,我们叫做协变,和string与Object的父子关系是相同的;反之,令List成为List子类的变化,我们叫做逆变...,和string与Object的父子关系是相反的; 协变和逆变能够实现数组类型、委托类型和泛型类型参数的隐式引用转换。

    13721

    C# | 泛型委托的协变与逆变

    协变 协变: 当类型参数仅用作输出(即返回值)时,可将类型参数标记为 out 这样就允许 Del dogMaker = MakeDog; // MakeDog是一个返回Dog匹配委托的方法...Console.WriteLine(animal().Leg.ToString()); Del 指向了 Del 调用代码 animal() 期望返回 Animal , 而其指向的...Del 返回了 Dog , Dog 是 Animal 的派生类,接受返回 Dog ,于是可以,调用代码可以自由的操作返回的对象的 Animal 部分 逆变 逆变: 当类型参数仅用作输入(作为方法的参数类型...Dog ,于是又传入了其指向了 Del , 于是传入 Del 了一个Animal的派生类Dog 这种在期望传入基类 时允许传入派生对象 的特性叫做逆变...本文作者: yiyun 本文链接: https://moeci.com/posts/2022/01/CSharp-泛型委托的协变与逆变/ 版权声明: 本博客所有文章除特别声明外,均采用 BY-NC-SA

    63920

    了解C#的协变和逆变

    前言 在引用类型系统时,协变、逆变和不变性具有如下定义。 这些示例假定一个名为 Base 的基类和一个名为 Derived的派生类。...以上来自于官方文档对协变、逆变、不变性的解释 为啥C#需要协变和逆变?...在C#中,目前只有泛型接口和泛型委托可以支持协变和逆变, 协变(Covariance) 内置的泛型协变接口,IEnumerator、IQuerable、IGrouping的输入参数 当要进行类型转换,占位符T要转换的目标类型也必须是其子类,上述例子则是FooBase转为Foo 总结 协变和逆变只对泛型委托和泛型接口有效,对普通的泛型类和泛型方法无效 协变和逆变的类型必须是引用类型...,因为值类型不具备继承性,因此类型转换存在不兼容性 泛型接口和泛型委托可同时存在协变和逆变的类型参数,即占位符T 参考 泛型中的协变和逆变 | Microsoft Docs 《你必须知道的.NET(第2

    94710

    Java泛型的协变和逆变

    背景 文接上回,说到过Java的泛型擦除问题,这块我又联想到一个有意思的考点泛型的协变和逆变。...相信大家也都见到过JDK中很多源码也有这么用,但为什么要这么写呢? 根因是Java的泛型没有协变类型,无法关联起来,也就没有关系。 利用通配符,?代表子类,T为父类。...逆变同样也是在各类源码中层出不穷,结合协变的理解,这块相信大家应该不难理解。 常见的例子比如,List的继承关系, 假设存在继承关系Object-》T、T-》A、T-》B; 即T为A、B的父类,协变面向子类;逆变面向Object,它是所有对象的父类。...小结 1、协变、逆变的区别要分清 2、另外,再提一点泛型和通配符的区别,当然也可以结合第一点理解, 用于定义泛型类和方法,擦除后为AAA类型; <?

    72600

    c# 协变和逆变的理解

    泛型类型参数支持协变和逆变,可在分配和使用泛型类型方面提供更大的灵活性。 2....数组也继承了这一特性,对于一个string[]类型而言 理解了上述概念后,让我们来看看协变和逆变的概念,这里我们只谈谈关于接口可变性中的一些内容。以下我简单给出一个接口及其实现。...但是当我们将上述代码的子类和父类的位置调换,换成上述注释中的代码,编译器则会报错。...也正是因此,为了防止开发者写出错误的代码,.net 设计者便用了协变和逆变(对应 out 和 in 关键字)来强制要求正确行为。...所以归根到底,协变和逆变只是一种约束而已,这种规范限制了你的泛型接口中要么只能有将类型参数当作返回值的协变相容方法(加了 out 关键字),要么只能有将类型参数当作输入值的逆变相容方法(加了 in 关键字

    1.5K60

    Java泛型的协变和逆变

    背景 文接上回,说到过Java的泛型擦除问题,这块我又联想到一个有意思的考点泛型的协变和逆变。...相信大家也都见到过JDK中很多源码也有这么用,但为什么要这么写呢? 根因是Java的泛型没有协变类型,无法关联起来,也就没有关系。 利用通配符,?代表子类,T为父类。...逆变同样也是在各类源码中层出不穷,结合协变的理解,这块相信大家应该不难理解。 常见的例子比如,List的继承关系, 假设存在继承关系Object-》T、T-》A、T-》B; 即T为A、B的父类,协变面向子类;逆变面向Object,它是所有对象的父类。...小结 1、协变、逆变的区别要分清。 2、另外,再提一点泛型和通配符的区别,当然也可以结合第一点理解, 用于定义泛型类和方法,擦除后为AAA类型; <?

    54350

    c#4.0中的不变(invariant)、协变(covariant)、逆变(contravariant)小记

    不变/协变/逆变,4.0中的这几个概念越念越象绕口令,如果单纯死记硬背,就算记住了,时间长了还是会忘记的。...园子里已经有不少高手撰文写过这个话题:比如“装配脑袋”的NET 4.0中的泛型协变和反变 (2008年他就已经搞明白了这个概念)、偶像Artech的“C# 4.0新特性-"协变"与"逆变"以及背后的编程思想...” 以及1-2-3的 协变(Covariance)和逆变(Contravariance)的十万个为什么 这里只是从应用的角度,简单记录一下: 从.net3.5开始,System命名空间里就定义了一个泛型委托...(儿子是人类,父母当然也是人类,不可能是畜生,呵) 这时,我们称T为逆变(ContraVariant)量,而TResult则为协变(CoVariant)量。...记忆方法:向上转型称协变(因为这种转型肯定是安全的,比较“和谐”),向下转型称逆变(因为不一定能转型成功,有出错的可能,称逆变) 最后:in,out这二个关键字不仅能用于泛型委托,同样也适用于泛型接口(

    862100

    Java基础知识:泛型的类型擦除、逆变与协变

    通过逆变,可以让泛型的约束变得更加宽松。 与协变不同,逆变放宽的是对父类的约束,而协变放宽的是对子类的约束。 但同样,逆变放宽类型约束是存在一定代价的: List list = new ArrayList(); list.get(0); //无法进行get 逆变与协变的使用场景: 当一个对象只作为泛型的生产者,也就是只取泛型的情况下...extends ; 例如 JDK 中 ArrayList 的集合构造法中就是使用了协变: public ArrayList(@NotNull @Flow(sourcelsContainer = true...super ; 例如 JDK 中 ArrayList 的 removeIf 方法就是使用了逆变: public boolean removeIf(Predicate<?...super ; 例如 JDK 中 ArrayList 的 removeIf 方法就是使用了逆变: public boolean removeIf(Predicate<?

    78030

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

    object> list = new List(); 这个为什么就可以 A:   这就要讲到C#泛型里的逆变协变了 Q:   细嗦细嗦 逆变协变 C#泛型中的逆变(in)协变(out)对于不常自定义泛型的开发来说..., 与协变相似, 需要在泛型 T 之前加上关键词 in 对比上方的协变, 逆变看起来就像是将基类赋值给子类, 但这其实符合里氏代换的 当我们调用 item.Print 时, 看起来允许传入的参数为 string..., 帮助开发者更好地复用代码, 同时通过约束限制可能会出现的破坏类型安全的操作 逆变协变的限制 虽然上面讲了逆变(in)协变(out)看起来是什么样的, 但我的那个朋友还是有些疑问 Q:   那我什么时候可以用逆变...A:   假设 IEnumerable 同时支持逆变协变, IEnumerable list = new List();进行赋值后, list中实际保存的类型是string...两者的限制简单总结就是 输入的用逆变 输出的用协变

    20020

    C#4.0新增功能03 泛型中的协变和逆变

    泛型类型参数支持协变和逆变,可在分配和使用泛型类型方面提供更大的灵活性。 在引用类型系统时,协变、逆变和不变性具有如下定义。...对于接口,协变类型参数可用作接口的方法的返回类型,而逆变类型参数可用作接口的方法的参数类型。 协变和逆变统称为“变体” 。 未标记为协变或逆变的泛型类型参数称为“固定参数” 。...泛型委托的类型参数中的协方差和逆变的效果类似于普通委托绑定中的协方差和逆变的效果(请参阅委托中的差异 (C#) 和委托中的差异 (Visual Basic))。...Visual Basic 和 C# 不允许违反协变和逆变类型参数的使用规则,也不允许将协变和逆变批注添加到接口和委托类型之外的类型参数中。...Variant 泛型接口和委托类型的列表 在 .NET Framework 4 中,下面的接口和委托类型具有协变和/或逆变类型参数。

    1.3K20

    C# 4.0新特性-协变与逆变以及背后的编程思想

    在《上篇》中我们揭示了“缺省参数”的本质,现在我们接着来谈谈C#4.0中另一个重要的新特性:协变(Covariance)与逆变(Contravariance)。...对于协变与逆变,大家肯定不会感到陌生,但是我相信有很多人不能很清晰地说出他们之间的区别。我希望通过这篇文章能够让读者更加深刻的认识协变与逆变。...目录 一、两个概念:强类型与弱类型 二、委托中的协变与逆变的使用 三、接口中的协变与逆变的使用 四、从Func...二、委托中的协变与逆变的使用 协变和逆变主要体现在两个地方:接口和委托,先来看看在委托中如何使用协变和逆变。...接下来我们来谈谈协变和逆变的本质区别是什么。

    63880

    五分钟看完,彻底理解C#的协变逆变

    协变、逆变 解决的问题 泛型类型转换 比如Person类是Student的父类,我们平时可以直接: Person A = new Student(); 这是所谓的隐式转换,相信百分之999.99%的人都知道...这种从派生类转向基类的兼容,就是所谓的协变。 说白了,我可以造个学生,结果你说给个人就行, 那不是so easy。...最后我们总结下,逆变和协变就是让方法有了泛型类型上的转换能力,强化了方法的多态能力。 问题点 1、属性为啥可以用逆变协变? 属性不就是get/set方法。...2、为什么接口和委托可以用逆变协变,类不行? 拜托你找一下共同点,接口和委托的共同点,都是行为,也就是方法为核心。接口里不能有字段。这也印证了我说的逆变协变最终是为方法服务的。...3、逆变和协变有啥用? 当你...设计问题,我就有遇到,有时候用上能更加优雅或者灵活的写代码吧,看你吧,少年。

    33020

    深入 TypeScript 中的子类型、逆变、协变,进阶 Vue3 源码前必须搞懂的。

    逆变和协变 先来段维基百科的定义: 协变与逆变(covariance and contravariance)是在计算机科学中,描述具有父/子型别关系的多个型别通过型别构造器、构造出的多个复杂型别之间是否有父...协变(Covariance) 那么想象一下,现在我们分别有这两个子类型的数组,他们之间的父子关系应该是怎么样的呢?...在 TS 中 当然,在 TypeScript 中,由于灵活性等权衡,对于函数参数默认的处理是 双向协变 的。...在开启了 tsconfig 中的 strictFunctionType 后才会严格按照 逆变 来约束赋值关系。...结语 这篇文章结合我自己最近学习类型相关知识的一些心得整理而成,如果有错误或者疏漏欢迎大家指出。 参考资料 Subsets & Subtypes TypeScript 官方文档 维基百科-协变与逆变

    1.3K31

    哲思片段 | 设计中的变与不变

    无论它在内存中存储的状态如何变化,该实例的对象标识依旧是保持不变的。显然,变与不变是相对的。 切换到DDD的命题中,所谓“实体”就是那种具有唯一的可识别可跟踪ID的对象。...与之相对的是值对象。在DDD中,强调将领域对象严格区分为实体和值对象。一个指导原则是,当你无法分辨某个领域对象究竟是实体还是值对象时,应优先将其建模为值对象。这有助于我们更好地利用值对象的不可变性。...这是赫拉克利特终极的哲学观,即万物随时在变。软件系统就是这样一条河流,它无时无刻不在变化,正如水不断的流动,需求也总是在变化。...这个Identity表达了单一、恒等的概念,例如Int类型中加减法运算半群(SemiGroup)中的Zero,就是一个Identity,因为半群中的任何元素a与Zero结合,依然是元素a本身。...水是如何组成一条河流的呢?这取决于组合子(Combinator)的设计与组合。只要我们找到万物的基本要素,继而设计出各种组合子,就可以演绎出世间不同的物。

    1.3K70
    领券