总结:
面试是一个不断学习、不断自我提升的过程,有机会还是出去面面,至少能想到查漏补缺效果,而且有些知识点,可能你自以为知道,但让你说,并不一定能说得很好。
有些东西有压力才有动力,而学到的知识点,都是钱(因为技术人员大部分情况是根据你的能力来定级、来发薪水的),技多不压身。
附上我的面试各大专题整理: 面试指南,满满的都是干货,希望对大家有帮助!
网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。
一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!
}
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)
}
控制台输出:
推荐学习资料
-
Android进阶学习全套手册
-
Android对标阿里P7学习视频
-
BAT TMD大厂Android高频面试题
网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。
一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!
-
Android进阶学习全套手册
[外链图片转存中…(img-5J2vMzl5-1715751927543)]
-
Android对标阿里P7学习视频
[外链图片转存中…(img-qzN1L6L2-1715751927543)]
-
BAT TMD大厂Android高频面试题
[外链图片转存中…(img-TvL4wBWn-1715751927544)]
网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。
一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!