Kotlin中,理解T.()->Unit 、 ()->Unit与
Kotlin比Java更方便的地方,其中之一是可以将函数作为参数。
上面三者都是将函数作为其它函数的参数来使用,其形式虽然简单,但理解并不简单。
一、共同点
三者的返回值相同,均为Unit,即没有返回值。
当然有返回值也可以,比如返回一个泛型R,或者是一个具体的值Int等
如:
T.()->R、 ()->R、(T) -> R
或者:
T.()->Int、 ()->Int、(T) -> Int
但是这不是重点,要理解三者,主要是分析前面的部分,这里返回Unit只是为了方便理解。
二、定义形式
可以结合Kotlin自带的作用域函数来理解:
apply:
fun T.apply(block: T.() -> Unit): T { block(); return this }
also:
fun T.also(block: (T) -> Unit): T { block(this); return this }
自定义一个使用()->Unit的例子:
inline fun <T> T.doWithTry(block: () -> Unit) {
try {
block()
} catch (e: Throwable) {
e.printStackTrace()
}
}
三、使用
//定义了一个Person类
class Person(val name:String){
var age:Int = 0
var sex:Int = 0
}
fun test() {
val person= Person("张三")
person.also {
//没有指定参数名字,必须用it代指参数
it.age = 20 //it不能省略
it.sex = 0 //it不能省略
}
//或者
person.also {personValue->
//使用指定的参数名,同样personValue不能省略
personValue.age = 20
personValue.sex = 0
}
person.apply {
//直接访问Person的属性
this.age = 20 //this可以省略
this.sex = 1 //this可以省略
}
person.doWithTry{
//只能通过外部变量来访问Person
person.age = 20
person.sex = 1
}
}
上面的例子说明了以下几点:
1、T.()->Unit 的函数体中可以直接使用T代表的对象,即用this代表对象
2、(T) -> Unit 将T表示的对象作为实参通过函数参数传递进来,供函数体使用
3、 ()->Unit与T表示的对象没有直接联系,只能通过外部T实例的变量来访问对象
四、理解
T.()->Unit 中用this代表对象,而this的使用一般是一个类的成员函数中用来表示该类的实例对象本身,比如,为Person类加一个函数:
class Person(val name:String){
var age:Int = 0
var sex:Int = 0
fun info(){
//此函数中用this表示了当前的Person对象,this可以省略
println("age is ${this.age},sex is ${if(sex == 0) "女" else "男"}")
}
}
所以我们可以这样理解,T.()->Unit相当于是给类T定义了一个扩展函数,该函数没有形参,没有返回值,当然我们也可以增加参数与返回值,道理是一样的。
正是因为T.()为T的扩展函数,所以可以在函数体里直接访问T对象的属性或者成员函数。
(T) -> Unit与 ()->Unit只是一个普通的函数,一个带有参数,类型为T,另一个没有参数而已。
(T) -> Unit在使用it表示实参,是Lambda表达式所规定
//Lambda表达式形式
person.also {
//没有指定参数名字,必须用it代指参数
it.age = 20 //it不能省略
it.sex = 0 //it不能省略
}
//不使用Lambda表达式的原始形式
person.also(
//定义了一个匿名函数,符合also的参数要求
//it为参数名,可以更改成其它任何符合要求的名称
fun(it:Person){
//由also函数的定义可知,person对象会通过形参it传给匿名函数
it.age = 20
it.sex = 0
}
)
五、总结
1、()->Unit与(T) -> Unit是相同的,只是一个带参,一个不带参
(T) -> Unit通过形参T可将对象作为实参传给函数,所以函数体里能通过it或者指定名称的方式来访问该对象
2、T.()->Unit 等同于为T定义了一个无参数的扩展函数,所以在函数体内可以直接通过this或省略来访问T代表的对象