【Kotlin篇】差异化分析,let,run,with,apply及also

本文详细介绍了Kotlin编程中的let、run、with和apply函数的使用场景、特性以及它们在对象操作、空安全检查和代码简洁性方面的应用,对比了let与run的区别,展示了设计模式学习资源的重要性。
摘要由CSDN通过智能技术生成

class Book() {

var name = “《数据结构》”

var price = 60

fun displayInfo() = print(“Book name : $name and price : $price”)

}

fun main(args: Array) {

val book = Book().let {

it.name = “《计算机网络》”

“This book is ${it.name}”

}

print(book)

}

控制台输出:

This book is 《计算机网络》

在上面案例中,我们对Book对象使用let作用域函数,在函数块的最后一句添加了一行字符串代码,并且对Book对象进行打印,我们可以看到最后控制台输出的结果为字符串“This book is 《计算机网络》”。

按照我们的编程思想,打印一个对象,输出必定是对象,但是使用let函数后,输出为最后一句字符串。这是由于let函数的特性导致。因为在Kotlin中,如果let块中的最后一条语句是非赋值语句,则默认情况下它是返回语句。

那如果我们将let块中最后一条语句修改为赋值语句,会发生什么变化?

fun main(args: Array) {

val book = Book().let {

it.name = “《计算机网络》”

}

print(book)

}

控制台输出:

kotlin.Unit

可以看到我们将Book对象的name值进行了赋值操作,同样对Book对象进行打印,但是最后控制台的输出结果为“kotlin.Unit”,这是因为在let函数块的最后一句是赋值语句,print则将其当做是一个函数来看待。

这是let角色设定的第一点:1️⃣

  • let块中的最后一条语句如果是非赋值语句,则默认情况下它是返回语句,反之,则返回的是一个 Unit类型

我们来看let第二点:2️⃣

  • let可用于空安全检查。

如需对非空对象执行操作,可对其使用安全调用操作符 ?. 并调用 let 在 lambda 表达式中执行操作。如下案例:

var name: String? = null

fun main(args: Array) {

val nameLength = name?.let {

it.length

} ?: “name为空时的值”

print(nameLength)

}

我们设置name为一个可空字符串,利用name?.let来进行空判断,只有当name不为空时,逻辑才能走进let函数块中。在这里,我们可能还看不出来let空判断的优势,但是当你有大量name的属性需要编写的时候,就能发现let的快速和简洁。

let第三点:3️⃣

  • let可对调用链的结果进行操作。

关于这一点,官方教程给出了一个案例,在这里就直接使用:

fun main(args: Array) {

val numbers = mutableListOf(“One”,“Two”,“Three”,“Four”,“Five”)

val resultsList = numbers.map { it.length }.filter { it > 3 }

print(resultsList)

}

我们的目的是获取数组列表中长度大于3的值。因为我们必须打印结果,所以我们将结果存储在一个单独的变量中,然后打印它。但是使用“let”操作符,我们可以将代码修改为:

fun main(args: Array) {

val numbers = mutableListOf(“One”,“Two”,“Three”,“Four”,“Five”)

numbers.map { it.length }.filter { it > 3 }.let {

print(it)

}

}

使用let后可以直接对数组列表中长度大于3的值进行打印,去掉了变量赋值这一步。

另外,let函数还存在一个特点。

let第四点:4️⃣

  • let可以将“It”重命名为一个可读的lambda参数。

let是通过使用“It”关键字来引用对象的上下文,因此,这个“It”可以被重命名为一个可读的lambda参数,如下将it重命名为book

fun main(args: Array) {

val book = Book().let {book ->

book.name = “《计算机网络》”

}

print(book)

}

2.2 run

run函数以“this”作为上下文对象,且它的调用方式与let一致。

另外,第一点:1️⃣** 当 lambda 表达式同时包含对象初始化和返回值的计算时,run更适合。**

这句话是什么意思?我们还是用案例来说话:

fun main(args: Array) {

Book().run {

name = “《计算机网络》”

price = 30

displayInfo()

}

}

控制台输出:

Book name : 《计算机网络》 and price : 30

如果不使用run函数,相同功能下代码会怎样?来看一看:

fun main(args: Array) {

val book = Book()

book.name = “《计算机网络》”

book.price = 30

book.displayInfo()

}

控制台输出:

Book name : 《计算机网络》 and price : 30

输出结果还是一样,但是run函数所带来的代码简洁程度已经显而易见。

除此之外,让我们来看看run函数的其他优点:

通过查看源码,了解到run函数存在两种声明方式

1、与let一样,run是作为T的扩展函数;

inline fun <T, R> T.run(block: T.() -> R): R

2、第二个run的声明方式则不同,它不是扩展函数,并且块中也没有输入值,因此,它不是用于传递对象并更改属性的类型,而是可以使你在需要表达式的地方就可以执行一个语句。

inline fun run(block: () -> R): R

如下利用run函数块执行方法,而不是作为一个扩展函数:

run {

val book = Book()

book.name = “《计算机网络》”

book.price = 30

book.displayInfo()

}

2.3 with

inline fun <T, R> with(receiver: T, block: T.() -> R): R

with属于非扩展函数,直接输入一个对象receiver,当输入receiver后,便可以更改receiver的属性,同时,它也与run做着同样的事情。

还是提供一个案例说明:

fun main(args: Array) {

val book = Book()

with(book) {

name = “《计算机网络》”

price = 40

}

print(book)

}

以上面为例,with(T)类型传入了一个参数book,则可以在with的代码块中访问book的name和price属性,并做更改。

with使用的是非null的对象,当函数块中不需要返回值时,可以使用with。

2.4 apply

inline fun T.apply(block: T.() -> Unit): T

apply是 T 的扩展函数,与run函数有些相似,它将对象的上下文引用为“this”而不是“it”,并且提供空安全检查,不同的是,apply不接受函数块中的返回值,返回的是自己的T类型对象。

fun main(args: Array) {

Book().apply {

name = “《计算机网络》”

price = 40

}

print(book)

}

控制台输出:

com.fuusy.kotlintest.Book@61bbe9ba

前面看到的 **let、with 和 run **函数返回的值都是 R。但是,apply 和下面查看的 also 返回 T。例如,在 let 中,没有在函数块中返回的值,最终会成为 Unit 类型,但在 apply 中,最后返回对象本身 (T) 时,它成为 Book 类型。

apply函数主要用于初始化或更改对象,因为它用于在不使用对象的函数的情况下返回自身

2.5 also

inline fun T.also(block: (T) -> Unit): T

自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数初中级Android工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则近万的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。

img

img

img

img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点,真正体系化!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!

如果你觉得这些内容对你有帮助,可以扫码获取!!(备注:Android)

设计模式学习笔记

设计模式系列学习视频

《互联网大厂面试真题解析、进阶开发核心学习笔记、全套讲解视频、实战项目源码讲义》点击传送门即可获取!

伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点,真正体系化!**

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!

如果你觉得这些内容对你有帮助,可以扫码获取!!(备注:Android)

设计模式学习笔记

[外链图片转存中…(img-l4W0hry0-1712831300156)]

设计模式系列学习视频

[外链图片转存中…(img-BbWiDAJn-1712831300157)]

《互联网大厂面试真题解析、进阶开发核心学习笔记、全套讲解视频、实战项目源码讲义》点击传送门即可获取!

  • 21
    点赞
  • 29
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值