Kotlin 范型之协变、逆变、不变

2024-09-06 05:28

本文主要是介绍Kotlin 范型之协变、逆变、不变,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

一. 前言

Kotlin 中类和类型是不一样的概念。型变是指类型转换后的继承关系。Kotlin 的型变分为逆变、协变和不变。

二. 协变

如果 A 是 B 的子类型,并且Generic<A> 也是 Generic<B> 的子类型,那么 Generic<T> 可以称之为一个协变类。

2.1Java 上界通配符<? extends T>

Java 的协变通过上界通配符实现。

如果 Dog 是 Animal 的子类,但 List<Dog> 并不是 List<Animal> 的子类。
下面的代码会在编译时报错:

 List<Animal> animals = new ArrayList<>();List<Dog> dogs = new ArrayList<>();animals = dogs; // incompatible types

而使用上界通配符之后,List<Dog> 变成了 List<? extends Animal> 的子类型。即 animals 变成了可以放入任何 Animal 及其子类的 List。

因此,下面的代码编译是正确的:

        List<? extends Animal> animals = new ArrayList<>();List<Dog> dogs = new ArrayList<>();animals = dogs;

2.2 Kotlin 的关键词 out

上述代码改成 Kotlin 的代码:

fun main() {var animals: List<Animal> = ArrayList()val dogs = ArrayList<Dog>()animals = dogs
}

居然没有编译报错?其实,Kotlin 的 List 跟 Java 的 List 并不一样。

Kotlin 的 List 源码中使用了outout相当于 Java 上界通配符。

public interface List<out E> : Collection<E> {override val size: Intoverride fun isEmpty(): Booleanoverride fun contains(element: @UnsafeVariance E): Booleanoverride fun iterator(): Iterator<E>override fun containsAll(elements: Collection<@UnsafeVariance E>): Booleanpublic operator fun get(index: Int): Epublic fun indexOf(element: @UnsafeVariance E): Intpublic fun lastIndexOf(element: @UnsafeVariance E): Intpublic fun listIterator(): ListIterator<E>public fun listIterator(index: Int): ListIterator<E>public fun subList(fromIndex: Int, toIndex: Int): List<E>
}

类的参数类型使用了out之后,该参数只能出现在方法的返回类型。

2.3 @UnsafeVariance

但是,Kotlin List 的 contains、containsAll、indexOf 和 lastIndexOf 方法中,入参均出现了范型 E。并且使用 @UnsafeVariance 修饰。

正是由于 @UnsafeVariance 的修饰,打破了刚才的限制,否则会编译报错。

三. 逆变

如果 A 是 B 的子类型,并且 Generic<B> 是 Generic<A> 的子类型,那么 Generic<T> 可以称之为一个逆变类。

3.1.1 Java 下界通配符<? super T>

Java 的逆变通过下界通配符实现。

下面的代码因为是协变的,无法添加新的对象。编译器只能知道类型是 Animal 的子类,并不能确定具体类型是什么,因此无法验证类型的安全性。

        List<? extends Animal> animals = new ArrayList<>();animals.add(new Dog()); // compile error

使用下界通配符之后,代码编译通过:

        List<? super Animal> animals = new ArrayList<>();animals.add(new Dog());

? super Animal 表示 Animal 及其父类 。所以 animals 可以接收所有 Animal 的子类添加至该列表中。

Java 的上界通配符和下界通配符符合 PECS 原则。

PECS 原则即 Producer Extends,Consumer Super 。如果参数化类型是一个生产者,则使用 <? extends T>;如果它是一个消费者,则使用 <? super T>。

其中,生产者表示频繁往外读取数据 T,而不从中添加数据。消费者表示只往里插入数据 T,而不读取数据。

3.1.2 Kotlin 的关键词 in

in相当于 Java 下界通配符。

abstract class Printer<in E> {abstract fun print(value: E): Unit
}class AnimalPrinter: Printer<Animal>() {override fun print(animal: Animal) {println("this is animal")}
}class DogPrinter : Printer<Dog>() {override fun print(dog: Dog) {println("this is dog")}
}fun main() {val animalPrinter = AnimalPrinter()animalPrinter.print(Animal())val dogPrinter = DogPrinter()dogPrinter.print(Dog())
}

类的参数类型使用了in之后,该参数只能出现在方法的入参。

四. 不变

默认情况下,Kotlin 中的泛型类是不变的。 这意味着它们既不是协变的也不是逆变的。

例如 MutableList,它可读可写,泛型没有使用inout

五. 总结

本文从 Kotlin 的类、类型引出了型变。介绍了 Kotlin 的协变、协变和不变的概念和特性,以及 Java 的上界通配符、下界通配符。

这篇关于Kotlin 范型之协变、逆变、不变的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1141122

相关文章

Android kotlin语言实现删除文件的解决方案

《Androidkotlin语言实现删除文件的解决方案》:本文主要介绍Androidkotlin语言实现删除文件的解决方案,在项目开发过程中,尤其是需要跨平台协作的项目,那么删除用户指定的文件的... 目录一、前言二、适用环境三、模板内容1.权限申请2.Activity中的模板一、前言在项目开发过程中,尤

Kotlin高阶函数与Lambda表达式及内联函数的介绍

目录 1、高阶函数1.1、什么是高阶函数?1.1.1、不带返回值的高阶函数1.1.2、带参数且带返回值的高阶函数1.1.3、与一般的函数进行比较 1.2、如何使用?1.3、高阶函数有什么作用? 2、Lambda表达式2.1、什么是Lambda表达式?2.1.1、无参数的写法2.1.2、有参数的写法2.1.3、有参数且有返回值的写法 2.2、如何使用?2.3、Lambda表达式有什么作用? 3

AI时代产品经理面临的变与不变:0经验求职产品经理要注意哪些细节?

AI时代,各种产品形态、业务的变化,让市场也对产品经理提出了新的要求,产品经理要有哪些变与不变呢?现在入行产品经理是好时机么?没有技术背景、没有学历有优势如何入行做产品经理?今天我们一起探讨一下! 产品人究竟需要具备哪些能力?看这个最新的能力模型图就知道了。 随着当前市场的细分,不同行业和领域对产品经理的能力要求已经从单一的具备产品专业能力演变成了兼具产品专业技能+行业/业务知识

android kotlin复习 Anonymous function 匿名函数

1、还是先上个图,新建kt: 2、代码: package com.jstonesoft.myapplication.testfun main(){val count = "helloworld".count()println(count);println("------------------------")var count2 = "helloworld".count(){it ==

android开发---Kotlin语言基础语法

目录 数据打印 变量 函数 程序逻辑控制   if  when 循环 数据打印 IDE采用的androidStudio 可自行官网下载 https://developer.android.google.cn/studio/archive?hl=zh-cn 新建项目 添加一个main方法,main()函数的左边出现了一个运行标志的小箭头。现在我们只要点击一下这个

有temp表包含A,B两列,使用SQL,对B列进行处理,形成C列,按A列顺序,B列值不变,则C列累计技术,B列值变化,则C列重新开始计数

有temp表,使用SQL,对B列进行处理,形成C列,按A列顺序,B列值不变,则C列累计技术,B列值变化,则C列重新开始计数 建表语句如下 CREATE TABLE temp(A STRING ,B STRING );INSERT INTO TABLE temp VALUES('2010','1'),('2011','1'),('2012','1'),('2013','0'),('2014',

《数字信号处理》学习04-离散时间系统中的线性时不变系统

目录 一,系统及离散时间系统   二,离散时间系统中的线性时不变系统 1,线性系统  1) 可加性  2) 比例性(齐次性) 3)叠加原理(叠加性质)  2,时不变系统(移不变系统) 通过前几篇文章的学习,此时我对序列的相关概念和运算已经有所掌握,接下来我将开始学习新的概念“离散时间系统中的线性时不变系统”, 一,系统及离散时间系统  首先需要知道系统的概念,在《信

裁剪视频如何让画质不变?小白都在用这些

怎么裁剪视频画面?其实,它已经成为许多人日常生活中的一项基本技能。 无论是制作温馨的家庭视频、分享日常点滴,还是进行专业的视频剪辑,裁剪视频都是一个关键步骤。 本文将为你详细介绍4个简单易学的视频裁剪教程,让你轻松掌握这项技能,让视频编辑变得不再高不可攀。 Part1选择正确的工具: ①剪辑魔法师:它支持多种视频格式,从MP4到MOV;且内置的丰富视频模板和特效素材,让你的视频瞬间提升

Object-Android关键字,伴生对象,Kotlin静态

目录 1、定义一个类并生成它的单例对象 原理 调用方式 2、伴生对象(静态类) 原理 调用方式 3、对象表达式 Object在Android独特的前缀通常有以下三种用法: 1、定义一个类并生成它的单例对象 object Singleton{fun test(){}} 原理 其实就是Java的单例模式,Kotlin中提供object方便创建单例 public fi

Map排序与转换的深入探索:Java与Kotlin的实现与应用

更多内容:孔乙己大叔 1. Map的排序基础 Map是一种键值对(Key-Value Pair)集合,其中每个键都映射到一个唯一的值。然而,Map接口本身并不保证顺序,这取决于具体的实现。例如,HashMap在Java和Kotlin中都是无序的,而TreeMap和LinkedHashMap则提供了有序的Map实现。 1.1 TreeMap排序 TreeMap在Java和Kotlin中都是基