Kotlin 函式魔法 - lambda 表示式

Mindjet發表於2017-10-26

lambda 表示式是指匿名函式的一種寫法,也可以拿來定義函式。

基本使用

  • lambda 表示式一般使用 {} 包裹
  • 一般格式為 param -> body

用於定義函式:

val foo = { str: String -> print(str) }

foo("xxx")複製程式碼

作為表示式直接使用:

fun main() {
    foo("xxx", { s -> s + "xxx" })
}

fun foo(a: String, b: (String) -> Unit) {
    b(a)
}複製程式碼

簡化寫法

  1. 當 lambda 表示式只接受一個引數時,該引數可以省略,使用時用 it 來表示

    foo("xxx", { s -> s + "xxx" })
    //等同於
    foo("xxx", { it + "xxx" })複製程式碼
  2. 當函式最後一個引數為函式時,該函式可以寫在 () 外,並用 {} 包裹

    foo("xxx", { s -> s + "xxx" })
    //等同於
    foo("xxx") { s -> s + "xxx" }
    //等同於
    foo("xxx") { it + "xxx" }複製程式碼
  3. 當函式只有一個引數,且該引數為函式時,可以直接省去 ()

    foo({ s -> s + "xxx" }) 
    //等同於
    foo { s -> s + "xxx" }複製程式碼
  4. 當引數在函式體中沒有引用時,可以將其設為 _,若此時只有一個引數(且該引數沒有被引用),則可以直接省略該引數;若有兩個或以上的引數,就算全部都沒有被引用,也不可以省略

    foo({ s -> print("xxx") })
    //等同於
    foo({ _ -> print("xxx") })
    //等同於
    foo({ print("xxx") })
    //等同於
    foo { print("xxx") }
    
    bar({ s, i -> print("xxx") })    //right
    bar({ _, _ -> print("xxx") })    //right
    bar({ print("xxx") })             //wrong, compile fail複製程式碼

在 Android 中的使用

在 Android 開發中,我們經常會遇到這樣一種回撥結構:

a.set(new B() {
      @override
    public void c(D d) {
        e(d);
    }
});
//實際上對我們有用的只是 e(d),B 或者 c 是什麼對我們不重要複製程式碼

比如:

handler.post(new Runnable() {
    @Override
    public void run() {

    }
});
//實際上對我們有用的只是 run 方法體內的語句

view.setOnClickListener(new View.OnClickListener() {
    @Override
    public void onClick(View v) {

    }
});
//實際上對我們有用的只是 onClick(View v) 方法體內的語句複製程式碼

此時,該結構並不是簡單的一個函式,而是一個實現了介面的匿名類或是一個匿名抽象類,不過 Kotlin 的語法特點也可以大大減少這種樣板程式碼的編寫,最終將這種結構簡化為對我們最有用的形式:

a.set(B({ d -> e(d) }))
//或
a.set(B{ d -> e(d) })

//當 B 的型別確定時(即非泛型),還可進一步簡化
a.set{ e(it) }
//我們確實只關注 e(),B 或者 c 是什麼對我們不重要複製程式碼

比如上面兩個例子可以簡化為:

handler.post(Runnable({ print("xxx") }))
//lambda表示式作為最後一個引數,可以將()去掉
handler.post(Runnable { print("xxx") })
//類的型別是確定的,可以將其去掉
handler.post({ print("xxx") })
//lambda表示式作為最後一個引數,可以將()去掉
handler.post { print("xxx") }

view.setOnClickListener(View.OnClickListener({ v -> print(v.id) }))
//lambda表示式作為最後一個引數,可以將()去掉
view.setOnClickListener { View.OnClickListener { v -> print(v.id) } }
//類的型別是確定的,可以將其去掉
view.setOnClickListener { v -> print(v.id) }
//只有一個引數,可以用it代替
view.setOnClickListener { print(it.id) }複製程式碼

是不是整個世界一下子清爽很多了呢?

趕緊在你的專案中用起來吧~!

———

技術上的問題,歡迎討論。

最近在 Github 上維護的專案:

  • LiteWeather [一款用 Kotlin 編寫,基於 MD 風格的輕量天氣 App],對使用 Kotlin 進行實際開發感興趣的同學可以看看,專案中會使用到 Kotlin 的委託機制、擴充套件機制和各種新奇的玩意。
  • LiteReader [一款基於 MD 的極輕閱讀 App,提供知乎日報、豆瓣電影等資源],專案主要使用了 MVVM 設計模式,介面遵循 Material Design 規範,提供輕量的閱讀體驗。
  • LiveMVVM [Kotlin 編寫的 Android MVVM 框架,基於 android-architecture],輕量 MVVM+Databinding 開發框架。
  • AnkoUtil [Kotlin 編寫的 Android 擴充套件庫]

歡迎 star/fork/follow 提 issue 和 PR。

相關文章