C#泛型接口的协变和逆变

 更新时间:2022年04月11日 11:50:47   作者:青草堂  
本文详细讲解了C#泛型接口的协变和逆变,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧

1、什么是协变、逆变?

假设:TSub是TParent的子类。
协变:如果一个泛型接口IFoo<T>,IFoo<TSub>可以转换为IFoo<TParent>的话,我们称这个过程为协变,IFoo支持对参数T的协变。
逆变:如果一个泛型接口IFoo<T>,IFoo<TParent>可以转换为IFoo<TSub>的话,我们称这个过程为逆变,IFoo支持对参数T的逆变。

2、为什么要有协变、逆变?

通常只有具备继承关系的对象才可以发生隐式类型转换,如Base b=new sub()。
协变和逆变可以使得更多的类型之间能够实现隐式类型转换、类型安全性有了保障。

3、为什么泛型接口要引入协变、逆变?

基于以上原因的同时、许多接口仅仅将类型参数用于参数或返回值。所以支持协变和逆变后泛型的使用上有了更大的灵活性

4、为什么支持协变的参数只能用于方法的返回值?支持逆变的参数只能用于方法参数?

“TParent不能安全转换成TSub”,是这两个问题的共同原因。
我们定义一个接口IFoo。

    interface IFoo<T>
    {
        void Method1(T param);
        T Method2();
    }

我们看一下协变的过程:IFoo<TSub>转换成IFoo<TParent>。

  • Method1:将TSub替换成TParent,Method1显然存在 TParent到TSub的转换。
  • Method2:返回值类型从TSub换成了TParent,是类型安全的。

所以支持协变的参数只能用在方法的返回值中。

再看一下逆变的过程:IFoo<TParent>转换成IFoo<TSub>。

  • Method1:将TParent替换成TSub,Method1存在 TSub到TParent的转换,是类型安全的。
  • Method2:返回值类型从TParent换成了TSub,是不安全的。

所以支持逆变的参数只能用在方法的参数中。

5、泛型接口支持协变、逆变和不支持协变、逆变的对比?

这其实是对3个问题的补充。

定义一个接口IFoo,既不支持协变,也不支持逆变。

    interface IFoo<T>
    {
        void Method1(T param);
        T Method2();
    }

实现接口IFoo

    public class FooClass<T> : IFoo<T>
    {
        public void Method1(T param)
        {
            Console.WriteLine(default(T));
        }
        public T Method2()
        {
            return default(T);
        }
    }

定义一个接口IBar支持对参数T的协变

    interface IBar<out T>
    {
        T Method();
    }

实现接口IBar

    public class BarClass<T> : IBar<T>
    {
        public T Method()
        {
            return default(T);
        }
    }

 定义一个接口IBaz支持对参数T的逆变

    interface IBaz<in T>
    {
        void Method(T param);
    }

实现接口IBaz

    public class BazClass<T> : IBaz<T>
    {
        public void Method(T param)
        {
            Console.WriteLine(param.ToString());
        }
    }

定义两个有继承关系的类型,IParent和SubClass。

    interface IParent
    {
        void DoSomething();
    }
    public class SubClass : IParent
    {
        public void DoSomething()
        {
            Console.WriteLine("SubMethod");
        }
    }

按照协变的逻辑,分别来使用IFoo和IBar。

            //IFoo 不支持对参数T的协变
            IFoo<SubClass> foo_sub = new FooClass<SubClass>();
            IFoo<IParent> foo_parent = foo_sub;//编译错误

            //IBar 支持对参数T的协变
            IBar<SubClass> bar_sub = new BarClass<SubClass>();
            IBar<IParent> bar_parent = bar_sub;

foo_parent = foo_sub 会提示编译时错误“无法将类型“IFoo<SubClass>”隐式转换为“IFoo<IParent>”。存在一个显式转换(是否缺少强制转换?)”

按照逆变的逻辑,分别来使用IFoo和IBaz。

            //IFoo 对参数T逆变不相容
            IFoo<IParent> foo_parent = null;
            IFoo<SubClass> foo_sub = foo_parent;//编译错误

            //IBaz 对参数T逆变相容
            IBaz<IParent> baz_parent = null;
            IBaz<SubClass> baz_sub = baz_parent;

 foo_sub = foo_parent 会提示编译时错误“无法将类型“IFoo<IParent>”隐式转换为“IFoo<ISub>”。存在一个显式转换(是否缺少强制转换?)”

6、.NET4.0对IEnumerable接口的修改?

2.0中的定义:

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

4.0中的定义:

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

可以看到4.0中增加了对协变的支持。

可以在两个版本试下, 下面的语句在2.0下会报错。

    List<SubClass> subarr = new List<SubClass>();
    IEnumerable<IParent> parentarr = subarr;

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。

相关文章

  • C#实现密码验证与输错密码账户锁定

    C#实现密码验证与输错密码账户锁定

    这篇文章介绍了C#实现密码验证与输错密码账户锁定的方法,文中通过示例代码介绍的非常详细。对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2022-04-04
  • C#中this的用法集锦

    C#中this的用法集锦

    本文给大家汇总介绍了C#中的几种this用法,相信大家应该有用过,但你用过几种?以下是个人总结的this几种用法,欢迎大家拍砖,废话少说,直接列出用法及相关代码。
    2015-06-06
  • C#的3DES加密解密算法实例代码

    C#的3DES加密解密算法实例代码

    这篇文章主要介绍了C#的3DES加密解密算法实例代码,有需要的朋友可以参考一下
    2013-11-11
  • C#实现将PDF转为Excel的方法详解

    C#实现将PDF转为Excel的方法详解

    通常,PDF格式的文档能支持的编辑功能不如office文档多,针对PDF文档里面有表格数据的,如果想要编辑表格里面的数据,可以将该PDF文档转为Excel格式。本文将介绍如何利用C#实现PDF转Excel,需要的可以参考一下
    2022-04-04
  • C# dll代码混淆加密的实现

    C# dll代码混淆加密的实现

    本文主要介绍了C# dll代码混淆加密的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2023-02-02
  • 深入理解C#中的扩展方法

    深入理解C#中的扩展方法

    下面这篇文章主要给大家介绍了关于c#中扩展方法的相关资料,文中通过示例代码介绍的非常详细,供大家学习参考,感兴趣的朋友可以了解下
    2020-06-06
  • 全面分析c# LINQ

    全面分析c# LINQ

    这篇文章主要介绍了c# LINQ的相关资料,帮助大家更好的理解和使用c#,感兴趣的朋友可以了解下
    2020-08-08
  • C#实现FTP传送文件的示例

    C#实现FTP传送文件的示例

    这篇文章主要介绍了C#实现FTP传送文件的示例,帮助大家更好的理解和学习c#的使用,感兴趣的朋友可以了解下
    2020-12-12
  • Unity绘制二维动态曲线

    Unity绘制二维动态曲线

    这篇文章主要为大家详细介绍了Unity绘制二维动态曲线,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2019-03-03
  • 一篇文章说通C#的属性Attribute

    一篇文章说通C#的属性Attribute

    这篇文章主要给大家介绍了如何通过一篇文章说通C#的属性Attribute,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2021-04-04

最新评论