当前位置: 首页 > news >正文

Kotlin方差

泛型类型简介

泛型类型提供了一种重用代码的方式,使数据类型可以作为变量使用。这样,我们就能创建可以操作不同类型对象的类、接口和方法,只要这些对象符合类型参数的要求。

在 Kotlin 中,泛型类型用 <T> 关键字定义,但 T 可以是任意字母或单词。

在本主题中,我们将深入探讨泛型编程,并介绍类型、子类型和变异性(variance)等概念。这些概念在像 Kotlin 这样的静态类型语言中非常重要,因为类型关系会在编译时检查,它们帮助保证类型安全,同时提供灵活性。


类型与子类型

  • 类型:定义了一组有效的值及其对应的一组适当且一致的操作。

  • 子类型:在层级关系中,子类型继承了超类型的所有特性(有效值和操作),但也可能添加额外的值或操作,或者对值进行限制。换句话说,子类型的每个值都是超类型的值,但反之不成立。
    这种关系通常通过面向对象编程中的继承实现。

举例:

val integer: Int = 1
val number: Number = integer // Int 是 Number 的子类型open class Animal
class Dog : Animal()fun main() {val dog: Dog = Dog()var animal: Animal = dog // Dog 是 Animal 的子类型
}

变异性(Variance)

当我们使用泛型(特别是基于泛型的集合)时,情况会变得复杂。

假设 IntNumber 的子类型,考虑泛型类 Box<T>

  • Box<Int> 是不是 Box<Number> 的子类型?

  • Box<Dog> 是不是 Box<Animal> 的子类型?
    Kotlin 中泛型默认是不变的(invariant),因此这些都不是子类型关系,会导致类型不匹配的错误:

class Box<T>val bd: Box<Animal> = Box<Dog>() // 错误
val bn: Box<Number> = Box<Int>() // 错误

这就是变异性的问题。


变异性的三种形式

  • 不变(Invariant):默认模式。对于两种不同类型 A 和 B,Class<A> 既不是 Class<B> 的子类型,也不是超类型。

  • 协变(Covariant):保留类型子关系方向。若 A 是 B 的子类型,则 Class<A> 也是 Class<B> 的子类型。通常适用于“输出”(只返回值)的位置,Kotlin 用 out 关键字表示。

  • 逆变(Contravariant):类型子关系反转。若 A 是 B 的子类型,则 Class<B>Class<A> 的子类型。通常适用于“输入”(只接收值)的位置,Kotlin 用 in 关键字表示。


不变示例(Invariant)

默认情况下泛型是不变的:

val mutableAnimalsFromDogs: MutableList<Animal> = mutableListOf<Dog>() // 错误
val boxOfAnimalsFromDogs: Box<Animal> = Box<Dog>() // 错误

因为既能读也能写,所以不允许这种类型转换。


协变示例(Covariance)

适用于只读场景,比如 Kotlin 的 List 是协变的:

val dogs: List<Dog> = listOf(Dog(), Dog())
val animals: List<Animal> = dogs // 允许,因为 List 是协变的(定义为 List<out T>)

如果定义自己的泛型类协变:

class Box<out T>val dogBox: Box<Dog> = Box<Dog>()
val animalBox: Box<Animal> = dogBox // 允许,Box 是协变的

注意:用 out 修饰后,只能用 T 作为返回类型,不能作为函数参数类型。


逆变示例(Contravariance)

适用于只写场景,比如比较器 Comparator

interface Comparator<in T> {fun compare(e1: T, e2: T): Int
}val animalComparator: Comparator<Animal> = ...
val dogComparator: Comparator<Dog> = animalComparator // 允许,Comparator 是逆变的

in 修饰后,只能用 T 作为函数参数类型,不能作为返回类型。


使用地点变异(Use-site variance)

通过在使用泛型时添加 outin 来实现变异:

fun copyAnimals(source: MutableList<out Animal>, destination: MutableList<in Animal>) {destination.addAll(source)
}val dogs: MutableList<Dog> = mutableListOf(Dog())
val animals: MutableList<Animal> = mutableListOf()copyAnimals(dogs, animals)

这里 source 是协变的,允许读取;destination 是逆变的,允许写入。


星号投影(Star Projection)

当你不知道具体泛型类型,或者不关心具体类型时,用 * 代替类型参数:

class Box<T>(val item: T)fun printItems(boxes: List<Box<*>>) {for (box in boxes) {println(box.item)}
}

这表示 Box 可以持有任何类型。


总结与注意事项

  • 默认泛型是不变的。

  • 只用作输出(返回值)的类型参数应声明为协变(out)。

  • 只用作输入(函数参数)的类型参数应声明为逆变(in)。

  • inout 的使用限制了类型参数在类中的使用方式,保证类型安全。

  • 错误使用可能导致运行时错误,需谨慎。


复杂示例

open class Animal {fun feed() = println("The animal is fed")
}class Dog : Animal() {fun pet() = println("The dog is petted")
}class Cat : Animal() {fun ignore() = println("The cat ignores you")
}class Box<in T, out R>(private var t: T, private val r: R) {fun put(t: T) {this.t = t}fun take(): R {return r}
}fun main() {val dogBox: Box<Animal, Dog> = Box(Dog(), Dog())dogBox.put(Cat())  // 可以,因为 Cat 是 Animal 的子类型val dog: Dog = dogBox.take()  // 返回 Dogval catBox: Box<Dog, Animal> = Box(Dog(), Cat())// catBox.put(Cat()) // 错误,Cat 不是 Dog 的子类型val animal: Animal = catBox.take()  // 返回 Animal
}
http://www.lryc.cn/news/593687.html

相关文章:

  • 1 渗透基础
  • ros2高级篇之高可用启动文件及配置编写
  • Spring AI 1.0版本 + 千问大模型之文本对话
  • node.js学习笔记1
  • 【数据类型与变量】
  • MySQL——约束类型
  • Springboot项目的搭建方式5种
  • 使用DataGrip连接安装在Linux上的Redis
  • Python+大模型 day02
  • 辛普森悖论
  • 使用看门狗实现复位
  • 1.初始化
  • Web开发 03
  • 双目摄像头品牌
  • 板子 5.29--7.19
  • 【科研绘图系列】R语言绘制显著性标记的热图
  • 【黄山派-SF32LB52】—硬件原理图学习笔记
  • 商业秘密视域下计算机软件的多重保护困境
  • 计算机网络:(十)虚拟专用网 VPN 和网络地址转换 NAT
  • Java多线程基础详解:从实现到线程安全
  • 6. 装饰器模式
  • ROS2 视频采集节点实现
  • Redis常见线上问题
  • 基于LSTM的时间序列到时间序列的回归模拟
  • Keepalived 监听服务切换与运维指南
  • C study notes[1]
  • C语言:20250719笔记
  • CentOS 清理技巧
  • 第二次总结(xss、js原型链)
  • 在开发板tmp目录下传输文件很快的原因和注意事项:重启开发板会清空tmp文件夹,记得复制文件到其他地方命令如下(cp 文件所在路径 文件要复制到的路径—)