Kotlin常用函数let、with、run、apply的用法和区别

在kotlin编程中let、with、run、apply这些函数使用率是非常高的,有时候可以通用,差别很小,但如果能记住他们的不同点,可以更加合理的选择使用。

在这之前首先要了解一下Lambda表达式的一些规则,这会帮助你理解使用这些函数的时候有没有( )可不可以用it代替参数等。因为这些函数的最后一个参数都是lambda。

如何理解lambda呢?可以把lambda理解为就是一个对象,但这个对象比较特殊,它是一段代码,既然是对象就可以作为函数的参数使用。这种对象称为函数对象。
lambda表达式的语法格式举例:
{ x: Int, y: Int -> x + y }
lambda 表达式是括在“{ }”中, “ ->” 前面为lambda参数,后面为函数体。为了方便理解用普通函数推导一下变化过程:

fun sum(x: Int, y: Int) = x + y

首先去掉了关键字fun和方法名:

(x: Int, y: Int) = x + y

之后分为了2部分,“=” 前面作为lambda参数,后面作为函数体,中间用“->” 代替:

(x: Int, y: Int) ->  x + y

最后去掉参数的小括号,在外面加上大括号:

{x: Int, y: Int ->  x + y}

最终得到的lambda表达式就可以作为参数使用了:

fun useLambda(a: Int, {x: Int, y: Int ->  x + y}){
  ...代码省略...
}

了解了基础的语法格式后,在来看下,lambda的几个重要规则:
Kotlin规则1:如果函数的最后一个参数是lambda,lambda 表达式可以放在圆括号之外(拖尾 lambda):

val result = useLambda(a: Int){x: Int, y: Int ->  x + y}

Kotlin规则2:如果函数只有lambda一个参数时候,圆括号可以省略:

修改了useLambda函数:
fun useLambda({str: String ->  str.length}){
  ...代码省略...
}
规则1改为:
val result = useLambda(){str: String ->  str.length}
规则2改为:
val result = useLambda{str: String ->  str.length}

Kotlin规则3:如果lambda表达式只有一个参数时候,不用声明参数,可以用隐式名称it代替参数:

val result = useLambda{it.length}

Kotlin规则4:lambda表达式中有参数未使用的时候,可用下划线“_” 代替

useLambda(){_, y ->  x * x}

了解了lambda表达式之后再来分析函数
let函数用法:

val result = obj.let {
   //TODO 使用it访问obj对象的方法或者属性
   it.xxx()
   //TODO 最后一行代码的结果为返回值赋值给result
}

从let函数写法和lambda的规则可以看出,它只有一个lambda参数,并且这个lambda的参数也只有一个,就是调用者对象本身。let最常用于判断对象空指针之后有连续操作的场景。举例:

obj?.let{
   it.dosomething1()
   it.dosomething2()
   it.dosomething3()
}

with函数用法:

val result = with(obj) {
//TODO 直接访问obj对象的方法或者属性,或者使用this.调用
//TODO 最后一行代码的结果为返回值赋值给result
}

同样我们可以看出,with函数有2个参数,第一个参数为要处理的对象,最后一个是lambda参数,根据规则1,放在了圆括号后边。在这个lambda表达式中提供了第一个参数的对象可以用this指定,也可以省略。

run函数用法:

val result = obj.run{
//TODO 直接访问obj对象的方法或者属性,或者使用this.调用
//TODO 最后一行代码的结果为返回值赋值给result
}

run函数只有一个lambda参数,run看上去是let与with的结合。因为run可以解决判空调用还可以去掉“it”解决let函数隐式名称调用问题。

apply函数用法:

val result = obj.apply{
//TODO 直接访问obj对象的方法或者属性,或者使用this.调用
}

可以看出与run函数几乎一样,唯一的区别是lambda中没有指定返回值。返回的是调用者对象本身,也就是obj。

总结:

函数 参数个数 内部访问上下文对象 返回值
let 1 it lambda最后一行代码返回值
with 2 this 或者 省略 lambda最后一行代码返回值
run 1 this 或者 省略 lambda最后一行代码返回值
apply 1 this 或者 省略 返回调用者对象
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章