0

0

C# 协变(Covariance)与逆变(Contravariance) - 泛型中的in和out关键字

幻夢星雲

幻夢星雲

发布时间:2025-12-25 15:42:08

|

581人浏览过

|

来源于php中文网

原创

协变(out)允许ienumerable视为ienumerable(b继承a),用于返回值;逆变(in)允许iconsumer赋给iconsumer,用于参数;二者仅适用于接口和委托,保障类型安全。

c# 协变(covariance)与逆变(contravariance) - 泛型中的in和out关键字

在 C# 中,协变(Covariance)和逆变(Contravariance)是泛型接口和委托中非常重要的概念,它们通过 outin 关键字实现,允许更灵活的类型转换。理解这两个特性有助于写出更具扩展性和复用性的代码。

什么是协变(Covariance)?

协变允许你使用比原本指定的类型更派生的类型。换句话说,如果类 B 继承自类 A,那么支持协变的泛型接口可以将 IEnumerable 视为 IEnumerable

协变通过在泛型类型参数前使用 out 关键字来声明。注意:只能用于返回值位置(如方法的返回类型),不能作为方法参数。

常见例子:

interface IProducer<out T>
{
    T Get();
}

这里 out T 表示 T 只能出现在输出位置(比如返回值)。因为只读,所以类型系统可以安全地进行向上转型。

实际应用:

string str = "hello";
IProducer<string> stringProducer = new StringProducer();
IProducer<object> objectProducer = stringProducer; // 协变支持

由于 string 是 object 的子类,且 IProducer 是协变的,这个赋值是合法的。

什么是逆变(Contravariance)?

逆变则相反:它允许你使用比原本指定的类型更基础的类型。也就是说,如果有一个接受基类的方法,它可以被用来处理派生类的对象。

AI发型设计
AI发型设计

虚拟发型试穿工具和发型模拟器

下载

逆变通过在泛型类型参数前使用 in 关键字来声明。它适用于输入参数,不能用于返回值。

例子:

interface IConsumer<in T>
{
    void Consume(T item);
}

这里 in T 表示 T 只能用作输入参数。因为方法只接收 T 类型对象,传入其子类对象是类型安全的。

实际使用:

IConsumer<object> objectConsumer = new ObjectConsumer();
IConsumer<string> stringConsumer = objectConsumer; // 逆变支持
stringConsumer.Consume("world"); // 实际调用的是 ObjectConsumer.Consume(object)

虽然 stringConsumer 声明为消费 string,但它指向的是一个能处理任意 object 的消费者,因此是安全的。

.NET 中常见的协变与逆变接口

微软在 .NET 框架中已经广泛使用了这些特性:

  • IEnumerable:协变 —— 允许将 IEnumerable 赋给 IEnumerable
  • IComparable:逆变 —— 如果一个类型能比较 Animal,那它当然也能比较 Dog
  • Action:逆变参数 —— Action 可以赋给 Action
  • Func:协变返回值 —— Func 可以当作 Func 使用

限制与注意事项

协变和逆变不是任意可用的,有严格的规则:

  • 只有接口和委托支持 inout,泛型类不支持
  • out T 只能出现在返回类型位置,不能作为方法参数
  • in T 只能出现在参数位置,不能作为返回类型
  • 数组也支持协变(如 string[] 可隐式转为 object[]),但这可能导致运行时异常(不安全),而泛型接口的协变是编译时安全的

基本上就这些。掌握 in 和 out 关键字,能让你更好地设计泛型接口,写出更自然、符合直觉的 API。关键是记住:out 用于产出,in 用于输入,就像数据流动的方向一样。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

930

2023.08.02

硬盘接口类型介绍
硬盘接口类型介绍

硬盘接口类型有IDE、SATA、SCSI、Fibre Channel、USB、eSATA、mSATA、PCIe等等。详细介绍:1、IDE接口是一种并行接口,主要用于连接硬盘和光驱等设备,它主要有两种类型:ATA和ATAPI,IDE接口已经逐渐被SATA接口;2、SATA接口是一种串行接口,相较于IDE接口,它具有更高的传输速度、更低的功耗和更小的体积;3、SCSI接口等等。

1800

2023.10.19

PHP接口编写教程
PHP接口编写教程

本专题整合了PHP接口编写教程,阅读专题下面的文章了解更多详细内容。

573

2025.10.17

php8.4实现接口限流的教程
php8.4实现接口限流的教程

PHP8.4本身不内置限流功能,需借助Redis(令牌桶)或Swoole(漏桶)实现;文件锁因I/O瓶颈、无跨机共享、秒级精度等缺陷不适用高并发场景。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

2341

2025.12.29

java接口相关教程
java接口相关教程

本专题整合了java接口相关内容,阅读专题下面的文章了解更多详细内容。

45

2026.01.19

C++类型转换方式
C++类型转换方式

本专题整合了C++类型转换相关内容,想了解更多相关内容,请阅读专题下面的文章。

317

2025.07.15

PHP高性能API设计与Laravel服务架构实践
PHP高性能API设计与Laravel服务架构实践

本专题围绕 PHP 在现代 Web 后端开发中的高性能实践展开,重点讲解基于 Laravel 框架构建可扩展 API 服务的核心方法。内容涵盖路由与中间件机制、服务容器与依赖注入、接口版本管理、缓存策略设计以及队列异步处理方案。同时结合高并发场景,深入分析性能瓶颈定位与优化思路,帮助开发者构建稳定、高效、易维护的 PHP 后端服务体系。

4

2026.03.04

AI安装教程大全
AI安装教程大全

2026最全AI工具安装教程专题:包含各版本AI绘图、AI视频、智能办公软件的本地化部署手册。全篇零基础友好,附带最新模型下载地址、一键安装脚本及常见报错修复方案。每日更新,收藏这一篇就够了,让AI安装不再报错!

7

2026.03.04

Swift iOS架构设计与MVVM模式实战
Swift iOS架构设计与MVVM模式实战

本专题聚焦 Swift 在 iOS 应用架构设计中的实践,系统讲解 MVVM 模式的核心思想、数据绑定机制、模块拆分策略以及组件化开发方法。内容涵盖网络层封装、状态管理、依赖注入与性能优化技巧。通过完整项目案例,帮助开发者构建结构清晰、可维护性强的 iOS 应用架构体系。

32

2026.03.03

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Go语言教程-全程干货无废话
Go语言教程-全程干货无废话

共100课时 | 11.2万人学习

Redis+MySQL数据库面试教程
Redis+MySQL数据库面试教程

共72课时 | 7万人学习

如何进行WebSocket调试
如何进行WebSocket调试

共1课时 | 0.1万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号