Kotlin中apply,run,let,also,with函数总结

函数介绍

apply

fun <T> T.apply(f: T.() -> Unit): T { f(); return this }

返回值是本身

run

run函数和apply函数很像,只不过run函数是使用最后一行的返回,apply返回当前自己的对象。

fun <T, R> T.run(f: T.() -> R): R = f()

let

默认当前这个对象作为闭包的it参数,返回值是函数里面最后一行,或者指定return

fun <T, R> T.let(f: (T) -> R): R = f(this)

let是将当前的函数传入后面的闭包中(it),可以随意指定返回值得类型 是不是意味着可以在let中,对其本身进行操作。而其他的不行呢?

also

从1.1新增的方法

public inline fun <T> T.also(block: (T) -> Unit): T { block(this); return this }

感觉是 apply 之于run 的,also至于let的存在。 都是在本身的调用方式上,强制返回本身

with

with函数是一个单独的函数,并不是Kotlin中的extension,所以调用方式有点不一样,返回是最后一行,然后可以直接调用对象的方法,感觉像是let和apply的结合。

fun <T, R> with(receiver: T, f: T.() -> R): R = receiver.f()

let难道不能指定返回自己吗?

总结

针对上面的疑问,用下面的代码进行了总结

    //apply
    m.apply {
        //1.调用函数可以直接写
        println(getMpValue())
        //2.如果调用外部函数对本身进行操作.通过this指代
        doChangeMp(this)
        println(getMpValue())
        //返回值强制就是本身
    }

    //run
    m.run {
        //1.调用函数直接写
        //1.调用函数可以直接写
        println(getMpValue())
        //2.如果调用外部函数对本身进行操作.通过this指代
        doChangeMp(this)
        println(getMpValue())
        //返回值可以自己定义.如果写this,就是和apply一模一样
        this
    }

    //let的三个点
    m.let {
        //1.调用是需要调用it,才能调用自身的方法。
        println(it.getMpValue())
        //2.调用函数对it进行操作
        doChangeMp(it)
        println(it.getMpValue())
        //3.最后一行返回值返回自己
        it
    }

    m.also {
        //1.调用是需要调用it,才能调用自身的方法。
        println(it.getMpValue())
        //2.调用函数对it进行操作
        doChangeMp(it)
        println(it.getMpValue())
        //返回值强制就是本身
    }
    
    //with
    //首先不是拓展函数。是单独的函数。其实with和run相似
    with(m){
        //调用方式和apply和run相似
        println(getMpValue())
        //2.如果调用外部函数对本身进行操作.通过this指代
        doChangeMp(this)
        println(getMpValue())
        //返回值可以自己定义.如果写this,就是和apply一模一样
        this
    }

不同

letalso因为使用it作为指代本身的关键字,就可以释放this关键字的使用了。 官网上的例子:

fun Block.copy() = Block().also { it.content = this.content }

其他

  • T.apply{}T.run{}其实是一样的。apply`只是将本身做了强制的返回。
  • T.also{}T.let{}其实是一样的。also只是将本身做了强制的返回。
  • with(T){}T.run{}其实是一致的。不同之处是,with是单独的函数。
  • 四者目前来看,没有什么不同。都能完成一样的工作。

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏猿人谷

《C++ primer》--第7章

删除指针后,该指针就变成了悬垂指针。悬垂指针指向曾经存放对象的内存,但该对象已经不再存在了。 习题7.8 举一个例子说明什么时候应该将形参定义为引用类型。再举一...

19450
来自专栏java达人

关于java传参

有一个比较经典的例子,如果你能说出此题的结果,说明你是java内存解析能手: public class Test1 { public static clas...

20470
来自专栏java一日一条

java提高篇之详解内部类

内部类是一个非常有用的特性但又比较难理解使用的特性(鄙人到现在都没有怎么使用过内部类,对内部类也只是略知一二)。

10420
来自专栏Java后端技术

java中带继承类的加载顺序详解及实战

  在面试中,在java基础方面,类的加载顺序经常被问及,很多时候我们是搞不清楚到底类的加载顺序是怎么样的,那么今天我们就来看看带有继承的类的加载顺序到底是怎么...

10720
来自专栏微信公众号:Java团长

Java基础11 对象引用

我们之前一直在使用“对象”这个概念,但没有探讨对象在内存中的具体存储方式。这方面的讨论将引出“对象引用”(object reference)这一重要概念。

8920
来自专栏纯洁的微笑

一个高频面试题,考考大家对 Java String 常量池的理解。

作为最基础的引用数据类型,Java 设计者为 String 提供了字符串常量池以提高其性能,那么字符串常量池的具体原理是什么,我们带着以下三个问题,去理解字符串...

21820
来自专栏开发与安全

从零开始学C++之构造函数与析构函数(一):构造函数、析构函数、赋值与初始化、explicit关键字

一、构造函数、默认构造函数 (1)、构造函数 构造函数是特殊的成员函数 创建类类型的新对象,系统自动会调用构造函数 构造函数是为了保证对象的每个数据成员都...

27600
来自专栏desperate633

深入理解javascript中的继承机制(1)原型链继承机制将共有的属性放进原型中

javascript中的继承机制是建立在原型的基础上的,所以必须先对原型有深刻的理解,笔者在之前已经写过关于js原型的文章。

8120
来自专栏机器学习从入门到成神

C++中重载、重写(覆盖)的区别实例分析

版权声明:本文为博主原创文章,未经博主允许不得转载。 https://blog.csdn.net/sinat_35512245/articl...

11630
来自专栏用户3030674的专栏

java线程的实现

一共有两种方法Thread类和Runnable接口,相对来讲,更趋向于用Runnable 因为一个类可以实现多个接口,但是只能继承一个类,所以相对来说倾向用Ru...

8410

扫码关注云+社区

领取腾讯云代金券