每日一题:请解释 .NET 中的协变和逆变?

请解释 .NET 中的"协变(Covariance)"和"逆变(Contravariance)"是什么?它们在泛型中的作用是什么?

参考答案:

在 .NET 中,协变(Covariance) 和 逆变(Contravariance) 是泛型类型参数的一种类型转换机制,用来提高代码的灵活性。它们主要用于 接口和委托的泛型类型参数。

协变(Covariance) 指的是:

如果类型 A 是 B 的子类型,那么 Generic<A> 可以被当作 Generic<B> 使用。

在 C# 中,协变通过关键字 out 表示,通常用于 只返回数据而不接收数据的类型参数。

例如:

如果 Cat 继承 Animal,那么 IEnumerable<Cat> 可以被当作 IEnumerable<Animal> 使用,因为集合只是读取数据,而不会写入数据。

逆变(Contravariance) 则是相反的方向:

如果 A 是 B 的子类型,那么 Generic<B> 可以被当作 Generic<A> 使用。

在 C# 中,逆变通过关键字 in 表示,通常用于 只接收参数而不返回数据的类型参数。

例如,在委托参数中,如果某个方法接受 Animal 参数,它也可以处理 Cat 参数。

协变和逆变的设计主要是为了 提高泛型类型的兼容性和可复用性,减少类型转换带来的限制。

追问 1

为什么协变只能用于返回值,而不能用于参数?

答案:

协变的核心思想是允许返回更具体的类型。如果允许在参数位置使用协变,就可能导致类型安全问题。例如,如果一个接口声明可以接收 Animal,但实际传入的是 Cat 的集合,那么调用方可能向集合中添加 Dog,这就会破坏原本 Cat 集合的类型安全。

因此,C# 语言设计中规定协变类型参数只能用于 输出位置(返回值),而不能用于输入位置(方法参数)。这样可以保证泛型类型在转换时仍然保持类型安全,同时又能提供一定程度的灵活性。

追问 2

在实际开发中,哪些 .NET 类型使用了协变或逆变?

答案:

在 .NET 标准库中,很多常用接口都使用了协变或逆变来提高灵活性。例如 IEnumerable<out T> 使用协变,因为集合通常只用于读取数据;而 IComparer<in T> 和 IEqualityComparer<in T> 使用逆变,因为它们主要用于接收参数进行比较。

此外,许多委托类型(例如 Func<> 和 Action<>)也使用了协变和逆变。例如 Func<out TResult> 的返回值支持协变,而 Action<in T> 的参数支持逆变。

这些设计使得 .NET 的泛型 API 更加灵活,减少了类型转换代码,同时保持类型安全。

#面试题 #dotnet面试题 #面试真题

相关推荐
小羊在睡觉10 小时前
Go与MySQL锁:索引失效陷阱
数据库·后端·mysql·golang
jiankeljx10 小时前
Spring Boot文件上传
java·spring boot·后端
cch891810 小时前
易语言 vs Go:初学者与专业开发之选
开发语言·后端·golang
0xDevNull10 小时前
Java 17 新特性概览与实战教程
java·开发语言·后端
Java成神之路-10 小时前
Spring IOC 注解进阶:@Bean 管理第三方 Bean,@Import 拆分配置,@Value 注入资源(Spring系列5)
java·后端·spring
zhenxin012210 小时前
Spring Data 什么是Spring Data 理解
java·后端·spring
dgvri10 小时前
Skywalking介绍,Skywalking 9.4 安装,SpringBoot集成Skywalking
spring boot·后端·skywalking
青柠代码录11 小时前
【SpringBoot】集成 Swagger
后端
1104.北光c°11 小时前
【重写优化 新增绘图】布谷鸟过滤器:布隆过滤器的更优缓存穿透解?
java·开发语言·后端·缓存·缓存穿透·布隆过滤器·布谷鸟过滤器
m0_6948455711 小时前
RevelGo搭建教程:类Rails开发体验的Go Web框架
服务器·开发语言·后端·docker·golang·开源·github