重学Kotlin(三) 扩展

65 阅读2分钟

本文已参与[新人创作礼]活动,一起开启掘金创作之路

持续创作,加速成长!这是我参与「掘金日新计划 · 10 月更文挑战」的第9天,点击查看活动详情

扩展函数

你可以为一个你不能修改的、来自第三方库中的类编写一个新的函数,可以用普通的方法调用。 这种机制称为 扩展函数 。 此外,也有 扩展属性 , 允许你为一个已经存在的类添加新的属性。

声明一个扩展函数,需要用被扩展的类型来作为他的前缀

//在函数名前声明泛型参数。
fun <T> MutableList<T>.swap(index1: Int, index2: Int) {
    val tmp = this[index1] // “this”对应该列表
    this[index1] = this[index2]
    this[index2] = tmp
}

这个 this 关键字对应到接收者对象(传过来的在点符号前的对象)

调用拓展函数:

val list = mutableListOf(1, 2, 3)
list.swap(0, 2) // “swap()”内部的“this”会保存“list”的值

扩展不能真正的修改他们所扩展的类。你并没有在一个类中插入新成员, 仅仅用点表达式去调用这个新函数相同函数情况下:成员函数优先级 > 拓展函数 相同函数 = 入参和函数名字都相同

可空接收者定义扩展,这样就可以在拓展函数内部判空

fun Any?.toString(): String {
    if (this == null) return "null"
    // 空检测之后,“this”会自动转换为非空类型,所以下面的 toString()
    // 解析为 Any 类的成员函数
    return toString()
}

扩展属性

//定义一个List末尾下标的拓展属性

//val <T> List<T>.lastIndex: Int = 0 错误写法,扩展属性不能有初始化器
val <T> List<T>.lastIndex: Int
    get() = size - 1

由于扩展没有实际的将成员插入类中,因此对扩展属性来说幕后字段是无效的。这就是为什么扩展属性不能有初始化器。他们的行为只能由显式提供的 getters/setters 定义。

//幕后字段使用field标识符在访问器中引用:
var counter = 0 // 注意:这个初始器直接为幕后字段赋值
    set(value) {
        if (value >= 0) field = value
    }

伴生对象的扩展

伴生对象定义扩展函数与属性

class MyClass {
    companion object { }  // 将被称为 "Companion"
}
//定义
fun MyClass.Companion.printCompanion() { println("companion") }
//调用
fun main() {
    MyClass.printCompanion()
}