本文已参与[新人创作礼]活动,一起开启掘金创作之路
持续创作,加速成长!这是我参与「掘金日新计划 · 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()
}