kotlin泛型

简介: kotlin泛型

泛型使用方法


在类上使用泛型

类上面使用泛型,需要把泛型声明在类名后,使用尖括号表示

class C1<T>{
    fun print(t:T){
        print(t.toString())
    }
}
复制代码


泛型函数

在fun关键字后使用声明

fun <T> method1(t:T):String{
    println(t.toString())
    return t.toString()
}
复制代码


型变


Java的型变

Java是不型变的,可以这么理解:

List<String> 并不是 List<Object>

举例如下代码是不可以编译通过的,因为无法型变

interface Collection<E> …… {
  void addAll(Collection<E> items);
}
void copyAll(Collection<Object> to, Collection<String> from) {
  to.addAll(from);
  // Collection<String> 不是 Collection<Object> 的子类型
}
复制代码


kotlin协变(out)

kotlin的协变使用out关键字表示,而java的协变用? extends E表示

如果一个类的泛型使用了协变<out T>,则T类型只能做为方法的返回值,不可以做为方法的形参。

举例:下面的代码中我们调用Factory.getCar最终获取的都是Car的类型,是安全的

open class Car{
}
class Taxi: Car() {
    fun drive(){
        println("开汽车")
    }
}
class Truck: Car() {
    fun drive(){
        println("开卡车")
    }
}
class Factory<out T>(private val car:T){
    fun getCar():T{
        return car
    }
//    fun setCar(t:T){//这个方法编译报错,因为T是协变的,所以只能获取(被消费),不能设置(被生产)
//        this.car=t
//    }
}
复制代码


kotlin逆变(in)

kotlin的逆变用in关键字表示,而java的逆变用? super E表示

同样用上面协变的代码举例,将T加上in修饰后,泛型就变为逆变的,此时Factory方法中getCar方法就会编译器报错,因为逆变的类型只能设置(被生产),不能获取(被消费)

open class Car{
}
class Taxi: Car() {
    fun drive(){
        println("开汽车")
    }
}
class Truck: Car() {
    fun drive(){
        println("开卡车")
    }
}
class Factory<in T>(private var car:T){
//    fun getCar():T{//Type parameter T is declared as 'in' but occurs in 'out' position in type T
//        return car
//    }
    fun setCar(t:T){
        this.car=t
    }
}
复制代码


上界


Java和Kotlin指定上界的方式

  • Java中泛型上界的表示方式:
Foo<T extend Noo>//表示泛型T是Noo的子类型
复制代码
  • Kotlin中泛型上界的表示方式
Foo<T : Noo>//表示泛型T是Noo的子类型
复制代码


Kotlin 方法中指定多个上界

kotlin可以使用where关键字实现方法中泛型指定多个上界 示例代码如下:

open class A{
}
interface B{
}
class C: A(),B{
}
class D: A() {
}
fun <T> copy(t:T,list:MutableList<T>) :List<A>
        where T:A,
              T:B{//where T:A,T:B 表示泛型T有两个上界A和B,其中A和B之中A是一个类B是接口,也就是说指定两个上界的时候只能有一个上界是类
    list.add(t)
    return list
}
fun main() {
    val list = mutableListOf<C>()
    copy(C(),list)//因为C同时是A和B的子类,所以这里编译可以通过
//    copy(D(),list)//因为D是A的子类,不是C的子类,所以这里编译器会直接提示错误
}



相关文章
|
4月前
|
安全 Java Kotlin
Kotlin泛型:灵活的类型参数化
Kotlin泛型:灵活的类型参数化
|
4月前
|
安全 Java 编译器
Android面试题之Java 泛型和Kotlin泛型
**Java泛型是JDK5引入的特性,用于编译时类型检查和安全。泛型擦除会在运行时移除类型参数,用Object或边界类型替换。这导致几个限制:不能直接创建泛型实例,不能使用instanceof,泛型数组与协变冲突,以及在静态上下文中的限制。通配符如<?>用于增强灵活性,<? extends T>只读,<? super T>只写。面试题涉及泛型原理和擦除机制。
34 3
Android面试题之Java 泛型和Kotlin泛型
|
Kotlin
Kotlin中接口、抽象类、泛型、out(协变)、in(逆变)、reified关键字的详解
Kotlin中接口、抽象类、泛型、out(协变)、in(逆变)、reified关键字的详解
86 0
|
安全 Java 编译器
Kotlin 泛型 VS Java 泛型
Kotlin 泛型 VS Java 泛型
67 0
|
存储 JSON 安全
From Java To Kotlin 2:Kotlin 类型系统与泛型终于懂了
上期主要分享了 From Java To Kotlin 1 :空安全、扩展、函数、Lambda。 这是 From Java to Kotlin 第二期。 带来 表达式思维、子类型化、类型系统、泛型。
205 0
From Java To Kotlin 2:Kotlin 类型系统与泛型终于懂了
|
安全 Java 编译器
Kotlin | 理解泛型使用
泛型,指的是具体的类型泛化,多用在集合中(如`List`、`Map`),编码时使用符号代替,在使用时再确定具体类型。
150 0
|
安全 Java C#
Kotlin 之泛型详解
Kotlin 之泛型详解
|
存储 安全 Java
「Java 路线」| 关于泛型能问的都在这里了(含Kotlin)
「Java 路线」| 关于泛型能问的都在这里了(含Kotlin)
129 0
「Java 路线」| 关于泛型能问的都在这里了(含Kotlin)
|
安全 Java 编译器
重学Kotlin之泛型的逆变和协变
泛型的逆变和协变
259 0
|
存储 Java 编译器
Kotlin | 浅谈 Reified 与泛型 的三两事
背景 在业务中,或者要写某个技术组件时,我们无可避免会经常使用到 泛型 ,从而让代码更具复用性与健壮性。 但相应的,由于Java泛型存在 类型擦除 的实现机制,所以某些情况下就会显得力不从心。而在 Kotlin 中,由于最终也会被编译为java字节码,所以无可避免也存在这上述问题🙂。
213 0