解析Kotlin中扩展函数与扩展属性【笔记摘要】

1.扩展函数

1.1 作用域:扩展函数写的位置不同,作用域就也不同

扩展函数可以写成顶层函数(Top-level Function),此时它只属于它所在的 package。这样你就能在任何类里使用它:

package com.rengwuxian

fun String.method1(i: Int) {
  ...
}
...
"rengwuxian".method1(1)

扩展函数也可以写在某个类里。然后你就可以在这个类里调用这个函数,但还是必须使用那个前缀类的对象来调用它:

class Example {
  fun String.method2(i: Int) {
    ...
  }
  ...
  "rengwuxian".method2(1) // 可以调用
}

注意:它属于Example的成员函数(限定该函数的调用范围),又属于String的扩展函数(限定该函数可以被谁调用)

1.2 指向扩展函数的引用

扩展函数如果是Top-Level的话,可以被引用;但是是某个类的成员函数,那就不可以被引用了:

fun String.method1(i: Int) {
    ...
}
...
String::method1

1.3 调用扩展函数的引用

直接调用或者用 invoke() 都可以,不过要记得把 Receiver 也就是接收者或者说调用者填成第一个参数

(String::method1)("rengwuxian", 1)
String::method1.invoke("rengwuxian", 1)

// 以上两句都等价于:
"rengwuxian".method1(1)

1.4 把扩展函数的引用赋值给变量

同样的,扩展函数的引用也可以赋值给变量:

val a: String.(Int) -> Unit = String::method1

然后你再拿着这个变量去调用,或者再次传递给别的变量,都是可以的:

"rengwuxian".a(1)
a("rengwuxian", 1)
a.invoke("rengwuxian", 1)

1.5 有无 Receiver 的变量的互换

class Example {
    fun happy(i: Int) {

    }
}
fun Example.sad(i: Int) {

}

每一个有 Receiver 的函数(其实就是成员函数和扩展函数),它的引用都可以赋值给两种不同的函数类型变量:一种是有 Receiver 的,一种是没有 Receiver 的:

val n: (Example, Int) -> Unit = Example::happy
val m: Example.(Int) -> Unit = Example::happy
var x: (Example, Int) -> Unit = Example::sad
var y: Example.(Int) -> Unit = Example::sad
x = m
y = n

1.6 改变原本的性质,使其可能无法被允许调用

一个普通的无 Receiver 的函数也可以直接赋值给有 Receiver 的变量:

fun mad(e: Example, i: Int) {

}
...

val e: (Example, Int) -> Unit = ::mad
val f: Example.(Int) -> Unit = ::mad

通过这些类型的互相转换,你可以把一个本来没有 Receiver 的函数变得可以通过 Receiver 来调用:
Example().mad(1)  //不允许调用,报错
Example().e(1)  //不允许调用,报错
Example().f(1)  //允许调用

当然了你也可以反向操作,去把一个有 Receiver 的函数变得不能用 Receiver 调用:

fun Example.sad(i: Int) {

}
...

val g: (Example, Int) -> Unit = Example::sad
val h: Example.(Int) -> Unit = Example::sad
Example().sad(1)  //允许调用
Example().g(1)  //不允许调用,报错
Example().h(1)  //允许调用

2.扩展属性

2.1 扩展变量与扩展常量

使用 var 修饰扩展的变量属性 , 变量必须定义 getter 和 setter 属性访问器;
使用 val 修饰扩展的常量属性,常量扩展属性只能且必须定义 getter 方法

2.2 定义位置

class Example {
    var name: String = "Tom"
    var age: Int = 18
}

和扩展函数一样,扩展属性可以定义在Top-level,也可以定义在某个类中
定义在Top-level,那么全局都可以使用该属性:

var Example.olderAge: Int
    get() {
        return this.age + 1
    }
    set(value) {
        this.age = value -1
    }

val Example.nameAndAge: String
    get() {
        return "${this.name} : ${this.age}"
    }

定义在某个类中,那么只能在该类中使用:

class Example {
    var name: String = "Tom"
    var age: Int = 18
    
    var Example.olderAge: Int
        get() {
            return this.age + 1
        }
        set(value) {
            this.age = value -1
        }

    val Example.nameAndAge: String
        get() {
            return "${this.name} : ${this.age}"
        }
}

2.3 扩展属性没有幕后字段

扩展属性由于没有幕后字段 , 因此不能定义属性的初始化器 ,给扩展属性赋初值会报以下错误:
在这里插入图片描述

扩展属性没有幕后字段不能保存实际的字段值 , 其属性访问器中只能调用对象中的属性和方法 , 不能调用扩展属性本身


参考文章:
会写「18.dp」只是个入门——Kotlin 的扩展函数和扩展属性(Extension Functions Properties)

  • 29
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值