Kotlin-let、run、also、apply区别

2021-06-12  本文已影响0人  杨0612
apply、also介绍
一看看apply、also源码
public inline fun <T> T.apply(block: T.() -> Unit): T {//1
    contract {
        callsInPlace(block, InvocationKind.EXACTLY_ONCE)
    }
    block()//2
    return this//返回值为this,也就是apply的调用者
}

public inline fun <T> T.also(block: (T) -> Unit): T {//3
    contract {
        callsInPlace(block, InvocationKind.EXACTLY_ONCE)
    }
    block(this)//4
    return this 返回值为this,也就是also的调用者
}
apply、also适用场景

因为返回值为调用者this,所以它们非常适合对同一个对象连续操作的链式调用。
以下代码以apply为例,链式调用对tv进行一系列操作。注意:例子不一定合理,只是想表达相应的意思而已。

    private fun init() {
        val tv = TextView(this)
        tv.apply {
            this.text = "name" //操作1
        }.apply {
            this.setOnClickListener { //操作2
                Log.d("MainActivity", "setOnClickListener")
            }
        }.apply {
            this.gravity = Gravity.CENTER //操作3
        }
    }
run、let介绍
一起看看 run、let源码
public inline fun <T, R> T.run(block: T.() -> R): R {//1
    contract {
        callsInPlace(block, InvocationKind.EXACTLY_ONCE)
    }
    return block()//2
}

public inline fun <T, R> T.let(block: (T) -> R): R {//3
    contract {
        callsInPlace(block, InvocationKind.EXACTLY_ONCE)
    }
    return block(this)//4
}
run、let适用场景

它们都可以有返回值,所以非常适合上一个操作返回值作用于下一个操作的链式调用。以下代码以let为例,操作1返回值作用于操作2,操作2返回值作用于操作3。注意:例子不一定合理,只是想表达相应的意思而已。

    private fun init(data: Int): Int {
        return data.let {
            if (data == 1) it + 1 else it + 2 //操作1
        }.let {
            if (data == 2) it + 3 else it + 4 //操作2
        }.let {
            if (data == 3) it + 5 else it + 6 //操作3
        }
    }
作用函数更重要的作用

确保操作的作用域,以下代码确保tv不为空的情况下执行,保证操作的作用域。

        val tv = TextView(this)
        tv?.apply {
            text = count.toString()
            setOnClickListener {
                Log.d("MainActivity", "setOnClickListener")
            }
            gravity = Gravity.CENTER
        }
为什么有的用this访问调用者,有的则用it?

前面分析源码的时候可以看到,

总结

以上分析有不对的地方,请指出,互相学习,谢谢哦!

上一篇 下一篇

猜你喜欢

热点阅读