美文网首页kotlin
Kotlin学习笔记(七)-泛型

Kotlin学习笔记(七)-泛型

作者: g小志 | 来源:发表于2019-12-18 19:29 被阅读0次

    [toc]

    前言

    这节我们说下Kotlin的泛型。首先默认大家对Java泛型有个基本的认识,如果 不熟悉Java的泛型,可以阅读文章,或是看下Java《Java核心技术卷一基础知识第10版》中关于泛型章节的知识,讲述的也很详细。其实Kotlin的泛型和Java很相似。他们都是伪泛型,所谓伪泛型就是我们们是无法获取到泛型的具体的类型的。以为Java存在类型擦除和转换。本篇还是和反射一样,从实际代码编写角度,说下Kotlin的泛型

    阅读本文前建议阅读文章:

    《Java核心技术卷一基础知识第10版》--泛型章节
    java 泛型详解

    泛型的协变与逆变
    Java 泛型之-协变&逆变

    逆变与协变

    泛型的逆变与协变其其在Java中也有。简单概括来说就是<? extends T>实现了泛型的协变,<? super T>实现了泛型的逆变。具体的这两种有什么特性可以看开头两篇文章。

    Kotlin中的协变与逆变

    • 泛型参数即可作为传入的参数,也可以作为返回值,但被in和out关键字修饰后就不一样了
    • out 叫协变 只能作为返回值读取 不能写入和修改(Kotlin的list中只有get方法 没有add)
    • in 叫逆变 只能写入不能读取 只能作为参数传入 (Kotlin中的Comparable只能传入参数)
    • 不变 既没有in也没有out就叫做不变 如MutableList(相当于java中的list)
    • 协变点:返回值类型是泛型类型参数
    • 逆变点:入参类型是泛型参数的类型
    • @UnsafeVariance 型变点伪例(当时协变的时候 泛型作为参数入参会报错 那么如果我们想忽略这个错误那么我就可以用这个注解标识)

    代码示例:

    协变 :泛型类型与实参的类型的继承关系相同
       val listOf: List<Number> = listOf<Int>(1, 2, 3)
    

    在泛型参数前面加上out表示协变,作为返回值,为只读类型,
    它的子类的泛型参数的类型是父类的泛型参数类型的子类,也就是说泛型参数的继承关系与类的继承关系保持一致(所以叫协变),比如Number是Int的父类 那么List<Number>也是List<Int>父类型;

    逆变 泛型参数的继承关系与类的继承关系相反
        val value: Comparable<Int> = object : Comparable<Any> {
            override fun compareTo(other: Any): Int {
                return 0
            }
        }
    

    在泛型参数前面加上in表示逆变,作为传入的参数,为只写类型,它的泛型参数的继承关系与类的继承关系相反,比如父类是Any,子类是Int。

    不变 类型必须保持一致 泛型之间没有关系
        val mutableList: MutableList<Int> = mutableListOf<Int>(1, 2, 3)
    
    星投影 其本质就是类似Java中通配符 ?
     val listOf1: List<Number> = listOf(1, 2, 3)
        //这是可以的
        val listOf2: List<*> = listOf(1, 2, 3)
    //    val listOf3: List<Number> = listOf<*>(1, 2, 3)//ERROR
    
        //这是可以的
        val value1: Comparable<*> = object : Comparable<Any> {
            override fun compareTo(other: Any): Int {
                return 0
            }
        }
        //可以
        val value2: Comparable<*> = object : Comparable<Int> {
            override fun compareTo(other: Int): Int {
                return 0
            }
        }
        //不可以
    //    val value3: Comparable<Int> = object : Comparable<*> {//error
    //        override fun compareTo(other: Int): Int {
    //            return 0
    //        }
    //    }
    
    //    val hello=Hello<*>//ERROR 因为泛型实参时不能用*代替
        //java 是可以有弱类型的(目的是兼容1.5) Kotlin不可以 定义了泛型了 创建的时候就必须指定泛型
    

    星投影只能只能作为形参,不能作为实参。

    reified 关键字

    reified单词含义为具体化的。用法

    inline fun <reified T> testGenerics2() {
        println(T::class.java)
    }
    

    加上reified关键字就可以打印出来他的类型了,这是在Java中做不到的。

    inline关键字的作用

    inline表示内联函数

    • inline 修饰符影响函数本身和传给它的 lambda 表达式:所有这些都将内联到调用处。
    • 内联可能导致生成的代码增加;不过如果我们使用得当(即避免内联过大函数),性能上会有所提升,尤其是在循环中的“超多态(megamorphic)”调用处。
    • inline关键字的作用是 一个方法带参数 同时这个方法中的参数是一个高阶函数(也就是Lambda表达式),那么inline可以提升性能

    结语

    其实泛型很难讲清楚。就算讲清楚了,也可能晦涩难懂。其实泛型掌握了编写规则。多实践,就好了。下篇讲下Kotlin的协程

    Github源码直接运行,包含全部详细笔记

    相关文章

      网友评论

        本文标题:Kotlin学习笔记(七)-泛型

        本文链接:https://www.haomeiwen.com/subject/auaunctx.html