美文网首页
Kotlin 学习笔记

Kotlin 学习笔记

作者: anap | 来源:发表于2020-08-25 11:44 被阅读0次

    1、Kotlin 中没有基础数据类型,都是用它的包装类型,他的包装类型是一个类,我们可以使用类里面很多有用的方法。

    2、kotlin访问修饰符

    访问修饰符可以修饰类,也可以修饰类的成员。

    (1)类访问修饰符

    (2)类成员访问修饰符

    有几点需要讲一下:

               1、 internal 修饰符是 Kotlin 独有而 Java 中没有的;

               2、protected 修饰符在Kotlin和Java中的异同点

                     · protected 修饰类,在 Java 中该类只能被同包名下的类访问,Kotlin 中就算是同包名的类也不能访问 protected 修饰的类。

                     · protected 修饰类成员,在 Java 中可以被同包名或子类可访问;在 Kotlin 中只能被子类访问。

    访问修饰符小结:

    (1)如果不写访问修饰符,在 Java 中是 default 修饰符 (package-private);在 Kotlin 中是 public 的

    (2)internal 访问修饰符是 Kotlin 独有,只能在模块内能访问的到

    (3)protected 修饰类的时候,不管是 Java 和 Kotlin 都只能放到内部类上

    (4)在 Kotlin 中,要继承 protected 类,要么子类在同一内部类名下;要么该类的的外部类和 protected 类的外部类有继承关系

    (5)在 Kotlin 中,继承 protected 类,子类也必须是 protected 的

    (6)在 Kotlin 中,对于 protected 修饰符,去掉了同包名能访问的特性

    (7)如果某个 Kotlin 类能够被继承,需要 open 关键字,默认是 final 的

    3、新建 bean 类的时候,常常需要声明 equals、hashCode、toString 等方法,我们需要写很多代码。在 Kotlin 中,只需要在声明类的时候前面加 data 关键字就可以完成这些功能。

    4、kotlin 新建内部类,例如:

    class OuterClass {

        //在 Kotlin 中内部类默认是静态的,不持有外部类的引用

        class InnerStaticClass{

        }

        //如果要声明非静态的内部类,需要加上 inner 关键字

        inner class InnerClass{

        }

    }

    编译后代码如下:

    class OuterClass {

      public static final class InnerStaticClass {

      }

      public final class InnerClass {

      }

    }

    5、companion object

    companion object 称之为伴生对象,伴生体里面放的是一些静态成员:如静态常量、静态变量、静态方法。例如

    companion object {

        //公有常量

        const val FEMALE: Int = 0

        const val MALE: Int = 1

        //私有常量

        val GENDER: Int = FEMALE

        //私有静态变量

        var username: String = "chiclaim"

        //静态方法

        fun run() {

            println("run...")

        }

    }

    6、Kotlin 中的 Interface

    在 Java8 之前,Interface 中是不能包含有方法体的方法和属性,只能包含抽象方法和常量。

    在 Kotlin 中的接口在定义的时候可以包含有方法体的方法,也可以包含属性。例如

    //声明一个接口,包含方法体的方法 plus 和一个属性 count

    interface InterfaceTest {

        var count: Int

        fun plus(num: Int) {

            count += num

        }

    }

    //实现该接口

    class Impl : InterfaceTest {

        //必须要覆盖 count 属性

        override var count: Int = 0

    }

    7、lambda 表达式

    在 Java8 之前,lambda 表达式在 Java 中都是没有的。

    (1)lambda 总是放在一个花括号里 ({})

    (2)箭头左边是 lambda 参数 (lambda parameter)

    (3)箭头右边是 lambda 体 (lambda body)

    8、高阶函数

    如果某个函数是以另一个函数作为参数或者返回值是一个函数,我们把这样的函数称之为高阶函数。

    例如:

    //Kotlin library filter function

    public inline fun <T> Iterable<T>.filter(predicate: (T) -> Boolean): List<T>

    //调用高阶函数 filter,直接传递 lambda 表达式

    list.filter { person ->

        person.age > 18

    }

    function type:

    名称 : (参数) -> 返回值类型

    (1)冒号左边是 function type 的名字

    (2)冒号右边是参数

    (3)尖括号右边是返回值

    比如:predicate: (T) -> Boolean predicate 就是名字,T 泛型就是参数,Boolean 就是返回值类型

    高阶函数是以另一个函数作为参数或者其返回值是一个函数,也可以说高阶函数参数是 function type 或者返回值是 function type

    在调用高阶函数的时候,我们可以传递 lambda,这是因为编译器会把 lambda 推导成 function type

    9、高阶函数 let、with、apply、run( 内联函数 ) 总结

    (1)let 函数一般用于判断是否为空

    //let 函数的定义

    public inline fun <T, R> T.let(block: (T) -> R): R {

        return block(this)

    }

    //let 的使用

    message?.let { //lambda参数it是message

        val result = it.substring(1)

        println(result)

    }

    (2) with 是全局函数,apply 是扩展函数,其他的都一样

    (3)run 函数的 lambda 是一个带有接受者的 lambda,而 let 不是,除此之外功能差不多

    10、Kotlin泛型

    Java 泛型:不变性 (invariance)、协变性 (covariance)、逆变性 (contravariance)

    Kotlin泛型:协变、逆变

    Kotlin 泛型擦除和具体化

    Kotlin 和 Java 的泛型只在编译时有效,运行时会被擦除 (type erasure)。例如下面的代码就会报错:

    //Error: Cannot check for instance of erased type: T

    //fun <T> isType(value: Any) = value is T

    Kotlin 提供了一种泛型具体化的技术,它的原理是这样的:

    我们知道泛型在运行时会擦除,但是在 inline 函数中我们可以指定泛型不被擦除,

    因为 inline 函数在编译期会 copy 到调用它的方法里,所以编译器会知道当前的方法中泛型对应的具体类型是什么,

    然后把泛型替换为具体类型,从而达到不被擦除的目的,在 inline 函数中我们可以通过 reified 关键字来标记这个泛型在编译时替换成具体类型

    如下面的代码就不会报错了:

    inline fun <reified T> isType(value: Any) = value is T

    11、Kotlin 集合

    Kotlin 中的集合底层也是使用 Java 集合框架那一套。在上层又封装了一层 可变集合 和 不可变集合 接口。

    (1)声明可变集合

    (2)声明不可变集合

    Kotlin 集合常用的 API

    1) all、any、count、find、firstOrNull、groupBy 函数

    2) filter、map、flatMap、flatten 函数

    延迟集合操作之 Sequences(Sequence就是就是为了避免创建多余的集合的问题。)

    val list = listOf<Person>(Person("chiclaim", 18), Person("yuzhiqiang", 15),

            Person("johnny", 27), Person("jackson", 190),

            Person("pony", 85))

    //把 filter 函数放置前面,可以有效减少 map 函数的调用次数

    list.asSequence().filter { person ->

        println("filter---> ${person.name} : ${person.age}")

        person.age > 20

    }.map { person ->

        println("map----> ${person.name} : ${person.age}")

        person.age

    }.forEach {

        println("---------符合条件的年龄 $it")

    }

    Sequence 的原理图如下所示:

    对 Sequence 做一个小结:

    (1)如果集合的数据量很大啊,可以使用集合操作的延迟 Sequence

    (2)Sequence 的 filter、map 等扩展还是是一个非 inline 的高阶函数

    (3)集合的 Sequence 只有调用 forEach、toList 等操作,才会触发对集合的操作。有点类似 RxJava。

    12、Koltin 和 Java 交互的一些问题

    1) Kotlin 和 Java 交互上关于空的问题

    2) 关于 Kotlin 基本类型初始化问题

    3) Kotlin 泛型具体化无法被 Java 调用

        如果我们定义了一个 inline 函数,且使用了泛型具体化,该方法不能被 Java 调用。反编译后发现该方法是私有的。只能Kotlin 代码自己调用。

    4) Kotlin 间接访问 Java default class

    相关文章

      网友评论

          本文标题:Kotlin 学习笔记

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