作用域函数,就是些技巧性方法,函数定义全在Standard.kt文件里。本篇,用实例记录一下它们的用法。

1.所谓作用域函数:它们的lamda函数作用域===对象类内部作用域。 就像是对象类的内部方法一样。

这样在lamda函数内,可以直接调用对象的方法,属性,不需要通过对象实例点出方法,属性。

2. let/takeIf/takeLess, 需要it引导

3. with是独立函数,其他函数都可由对象点出来。

  • run,利用操作对象的方法属性,完成一些工作。最后在函数体内返回指定值。
val list = listOf("apple", "orange", "pear")
        val result = StringBuilder().run {
            append("Start eating fruits.\n")
            for (fruit in list){
                append(fruit).append("\n")
            }

            append("Ate all fruits.")

            toString()
        }

        println(result)
  • apply,用于修改对象自己。修改后,如果函数有接收返回,接收的也是修改后的自己。函数体类不需要返回对象。
fun main() {
    val user = User("alan", 18)
    val applyResult = user.apply {
        //这儿都是user对象的作用域
        age = 20
    }
    println(applyResult) //输出:User(name=Peter, age=20)
    println(user) //输出:User(name=Peter, age=20)
    println(applyResult === user) //输出:true
}

data class User(var name: String, var age: Int)

第二个例子:

val list = listOf("apple", "orange", "pear")
        val result = StringBuilder().apply{
            append("Start eating fruits.\n")
            for (fruit in list){
                append(fruit).append("\n")
            }

            append("Ate all fruits.")
        }

        println(result.toString())
  • with,作用和run一样。不同的是,run由对象点出,而with是独立的函数,对象是作为with的参数。
val list = listOf("apple", "orange", "pear")
        val result = with(StringBuilder()) {
            append("Start eating fruits.\n")
            for (fruit in list){
                append(fruit).append("\n")
            }

            append("Ate all fruits.")

            toString()
        }

        println(result)
  • let, 把原始对象作为lamda表达式的参数,传入函数体。通常用来简化判空操作, 而且是线程安全的。
object.let{
   it.todo()//在函数体内使用it替代object对象去访问其公有的属性和方法
   ...
}

//另一种用途 判断object为null的操作
object?.let{//表示object不为null的条件下,才会去执行let函数体
   it.todo()
}

 下面的写法,有点罗嗦,如果使用user的参数方法更多,就要被问号烦死了。

fun doStudy(user:User?){
        user?.age = 20
        user?.name = "alan"
    }

或者用if判空:

fun doStudy(user:User?){
        if(user != null){
           user.age = 20
           user.name = "alan"
        }
    }

用let函数简化:

fun doStudy(user:User?){
        user?.let{
        it.age = 20
        it.name = "alan"
    }

针对if判空那种方式,是有问题的。当User对象是全局对象的时候。就编译错误了。提示有多线程问题。

var user:User? = null
    fun doStudy() {
        if (user!=null) {
            user.age = 20 //编译错误提醒
            user.name = "alan"//编译错误提醒
        }
    }
  • also, 和let的作用一样。不同点就是,let函数的返回值是最后一行的返回值,而also函数的返回值是返回上下文对象本身。可以理解为"并且为该对象执行以下操作"。

一般可用于多个扩展函数链式调用。

fun main() {
    val user = User("alan", 18)
    val alsoResult = user.also {
        println("user 之前的名字是:${it.name}") //user 之前的名字是:alan
    }.apply {
        name = "lin"
    }
    println(alsoResult) //User(name=lin, age=18)
    println(user) //User(name=lin, age=18)
    println(alsoResult === user) //true
}

data class User(var name: String, var age: Int)
  • takeIf,takeunless, 前面调用了一个函数计算得出了一个结果,现在需要对这个结果做一个分支判断,并且我们只需要用到if的一个分支时,可以用takeIf和takeUnless代替。

takeIf,takeunless, 区别就只有一个是满足代码块里面的条件才返回对象本身,一个是不满足条件才返回。

下面三段代码都是等价的:

用if语句:

fun testWithoutTakeIf() {
        val name = "alan Gong"
        val hasGong = name.indexOf("Gong")
        Log.i(TAG, "testWithoutTakeIf: hasGong = $hasGong")
        if (hasGong >= 0) {
            Log.i(TAG, "testWithoutTakeIf: has Gong")
        }
        Log.i(TAG, "testWithoutTakeIf: $name")
    }
    
输出:
I: testWithoutTakeIf: hasGong = 0
I: testWithoutTakeIf: has Gong 
I: testWithoutTakeIf: alan Gong

用takeIf:

fun testTakeIf() {
        val name = "alan Gong"
        name.indexOf("Gong")
            .takeIf {
                Log.i(TAG, "testTakeIf: it = $it")
                it >= 0
            }
            ?.let {
                Log.i(TAG, "testTakeIf: has Gong")
            }
        Log.i(TAG, "testTakeIf: $name")
    }
   
输出:
I: testTakeIf: it = 0
I: testTakeIf: has Gong
I: testTakeIf: alan Gong

用takeLess: 注意函数体的判断和takeIf函数体里相反。

fun testTakeUnless() {
        val name = "alan Gong"
        name.indexOf("Gong")
            .takeUnless {
                Log.i(TAG, "testTakeUnless: it = $it")
                it < 0
            }
            ?.let {
                Log.i(TAG, "testTakeUnless: has Gong")
            }
        Log.i(TAG, "testTakeUnless: $name")
    }
    
输出:
I: testTakeUnless: it = 0
I: testTakeUnless: has  Gong
I: testTakeUnless: alan Gong

由上面的简单例子也可以看出,这两个通常和其他作用域函数,组成链式调用。

  • repeat, 循环执行n次block中的代码。
repeat(3){
    println("repeat")
}

好了,目前所有的作用域函数都记录清楚了。看起来是挺强大的,但是这种东西多了,可读性就不见得好了。另外还有一张网络上的总结图:

Android kotlin copy撖寡情 kotlin的apply_函数体