重学Kotlin(三) 扩展

扩展函数

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

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

//在函数名前声明泛型参数。
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()
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

哆啦A梦z

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值