kotlin 在 standard.kt 文件中 提供好幾個(gè)挺有趣的方法。用的好的話秕脓,會(huì)讓我們代碼可讀性更強(qiáng),更加簡(jiǎn)潔儒搭。
昨晚看到一篇文章講這幾個(gè)方法吠架,講的非常到位。 《Mastering Kotlin standard functions: run, with, let, also and apply》
至于什么時(shí)候選擇用哪一個(gè)方法的時(shí)候师妙,可以上一張總結(jié)的圖:
T.run
先來看看定義:
run
其實(shí)有兩個(gè)定義:
* 接受一個(gè) block诵肛,返回block執(zhí)行結(jié)果,這個(gè)block里面捕獲到的'this'是block之外的作用域.
* T.run 同樣也是接受一個(gè)block,只是這個(gè)block是屬于T的一個(gè)`擴(kuò)展` 怔檩,返回block執(zhí)行結(jié)果褪秀,這個(gè)block里面捕獲到的'this'是`T`.
如何使用:
fun test() {
var mood = "I am sad"
run {
// 這里捕獲到的this是外層的作用域
val mood = "I am happy"
println(mood) // I am happy
}
println(mood) // I am sad
}
這種寫法,我們可以在任意位置薛训,新起一個(gè)代碼塊媒吗,而且可以訪問到上下文的,最終運(yùn)算出來一個(gè)結(jié)果值,可以讓代碼看起來更簡(jiǎn)潔乙埃, 比如:
run {
if (firstTimeView) introView else normalView
}.show()
為了區(qū)分一下兩個(gè)run闸英,再舉個(gè)例子:
T.with
先來看看定義:
接受一個(gè)T,和一個(gè)T的擴(kuò)展block介袜,返回block執(zhí)行的結(jié)果值
如何使用:
//with(webview.settings,{...}) 接受了一個(gè)setting對(duì)象和一個(gè)block.
// kotlin可以簡(jiǎn)寫把block直接跟著方法末端(尾閉包的概念)
with(webview.settings) {
//這里可以對(duì)T也就是settings進(jìn)行初始化操作
javaScriptEnabled = true
databaseEnabled = true
}
T.let
先來看看定義:
接受一個(gè)T為參數(shù)的block甫何,返回block執(zhí)行的結(jié)果值
如何使用:
stringVariable?.run {
println("The length of this String is $length")
}
// Similarly.
stringVariable?.let {
println("The length of this String is ${it.length}")
}
可以看到 let 和 run 非常相似,他們都可以做到接受前面的T遇伞,作為block的參數(shù)辙喂,但是let在這一塊的做法更為直接明確一些,因?yàn)門在block有自己的別名了的鸠珠,叫做'it'巍耗,而這里的this
就類似普通的 run 方法,可以訪問到上下文渐排。所以炬太,let有一個(gè)小小的用法,類似這樣:
stringVariable?.let {
nonNullString ->
println("The non null string is $nonNullString")
}
什么意思呢驯耻?
大家都知道亲族,kotlin中的?
的用法,也就是當(dāng)遇到前面一個(gè)為null的時(shí)候吓歇,后面將不會(huì)執(zhí)行孽水。所以上面的let用法就是在解包
,當(dāng)一個(gè)對(duì)象存在的情況的時(shí)候城看,做一系列相關(guān)操作女气。
T.also
先來看看定義:
接受一個(gè)T為參數(shù)的block,執(zhí)行完block后测柠,返回self炼鞠。可以看到 also 和前面提到的run let with都不同轰胁,
他的block是沒有返回值的:block: (T) -> Unit谒主。
如何使用:
// Normal approach
fun makeDir(path: String): File {
val result = File(path)
result.mkdirs()
return result
}
// Improved approach
fun makeDir(path: String) = path.let{ File(it) }.also{ it.mkdirs() }
also 的block 拿到的參數(shù)是前面let吐出來的File對(duì)象,所以also可以直接去創(chuàng)建文件赃阀,是不是比我們?cè)跊]改造之前更加簡(jiǎn)潔呢霎肯?
T.apply
先來看看定義:
接受一個(gè)T的擴(kuò)展block,執(zhí)行完block后,返回self观游。和 前面的also非常相似搂捧,但是不同的是apply接受的block是自己的擴(kuò)展。由于是擴(kuò)展block懂缕,即要拿到自己需要用this或者省略允跑。而also閉包里面前面的T是當(dāng)參數(shù)傳入給了閉包,所以此時(shí)有了參數(shù)別名`it`搪柑。
如何使用:
// Normal approach
fun createInstance(args: Bundle) : MyFragment {
val fragment = MyFragment()
fragment.arguments = args
return fragment
}
// Improved approach
fun createInstance(args: Bundle)
= MyFragment().apply { arguments = args }
正如我剛才解釋中提到的聋丝,他和 also 非常相似。