作者:Antonio Leiva
時間:Jan 5, 2017
原文鏈接:https://antonioleiva.com/lambdas-kotlin/
?
?
由于Lambda表達式允許更簡單的方式建模式函數,所以它是Kotlin和任何其他現代開發語言的最強工具之一。
?
在Java6中,我們僅能下面方法這樣做:通過用單一方法聲明接口,并用匿名對象實現那些接口。
?
Lambda表達式特別是在Kotlin中能定義的方式,為我們開啟無線可能的世界。在接下來的內容中,我們將了解這些用法。
?
Kotlin的Lambda表達式
?
Lambda表達式是函數的表示的方式,在解釋setOnClickListener時,我們已經見過這個例子:
1 val view = findViewById(R.id.welcomeMessage) 2 view.setOnClickListener { v -> navigateWithView(v) }
?
如你所見,左邊定義函數(在這個例子中時view)的輸入值,右邊聲明函數要實現的操作。
?
怎樣定義接受Lambda表達式的函數
?
如果我們自己要在Kotlin中定義函數,我們就需要按如下做法:
1 fun setOnClickListener(listener: (view: View) -> Unit){}
?
由于這個函數通過參數接收一函數,或返回一函數,所以,這是一高階函數。
?
Kotlin和Java interop
?
調用這個函數的一般方式可能是如下:
1 view.setOnClickListener({ v -> navigateWithView(v) })
?
這是我們已經見到這樣做的比較簡單的方法,而后續我們還能知道它幫助我們做更酷的事。
?
?
這是由于如果函數的最后一個參數是函數,我們能夠將它從括號中提取出來:
1 view.setOnClickListener(){ v -> navigateWithView(v) }
?
另外,如果只有一個函數作為參數,我們還能省去括號:
1 view.setOnClickListener { v -> navigateWithView(v) }
?
DSL創建
?
這允許我們創建自己的DSL,這能定義微型語言。在Kotlin參考網站有HTML例子,而這里我們要定義一個更簡介方法。
?
?
假設你要創建在另一個線程上運行的代碼塊。你可以有一個函數,它接收在后臺運行的一個函數:
1 fun doAsync(f: () -> Unit) { 2 Thread({ f() }).start() 3 }
?
這個函數產生一個線程,它執行Runnable運行作為自變量接受的函數。Runnable是Java中有單一方法的類,在Kotlin中由Lambda表達式替代。
?
?
現在,在我們的代碼中,產生異步代碼塊:
1 doAsync { 2 op1() 3 op2() 4 op3() 5 }
?
這樣在{}內的每件事都將在第二個線程中執行。
?
?
內嵌函數
?
作為自變量接收的函數令人討厭的是編譯器需要為它們創建類,這將影響性能。但是,這可以用保留字inline容易地解決。
?
?
由于Inline函數是在編譯時將它的代碼替換對它的調用,所以在性能方面影響較少。為此,它不需要用額為的對象。
?
?
我們能夠轉換doAsync到一個inline函數:
1 inline fun doAsync(crossinline f: () -> Unit) { 2 Thread({ f() }).start() 3 }
?
在這個例子中,由于我們從另一個執行內容(另一個Lambda表達式)中調用f(),所以要求crossinline。不用太擔心這些,在需要使用它時,編譯器會提醒你的。
?
?
結論
?
如你所見,使用Lambda表達式后,我們能夠簡化很多我們的代碼,甚至是在Java中不能實現的事。
?
?
另外,Kotlin的特別命名規則使我們能夠創建屬于我們自己的“開發語言”,并且還可以根據需要創建有意義的代碼塊。
?
?
Lambda表達式功能非常強大,這本書包含許多你能使用它們不同的情況。
?
Kotlin和Java interop