玩轉(zhuǎn)Kotlin 徹底弄懂Lambda和高階函數(shù)
Lambda是什么
簡單來講,Lambda是一種函數(shù)的表示方式(言外之意也就是說一個Lambda表達(dá)式等于一個函數(shù))。更確切的說:Lambda是一個未聲明的函數(shù),會以表達(dá)式的形式傳遞
為什么要用Lambda
設(shè)想一下,在Android中實(shí)現(xiàn)一個View的點(diǎn)擊事件,可以使用如下實(shí)現(xiàn):
View view = findViewById(R.id.textView); view.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View view) { viewClicked(view); } });
而如果在Kotlin中使用Lambda,則實(shí)現(xiàn)可以簡單如下:
val view = findViewById(R.id.image) view.setOnClickListener { v -> viewClicked(v) }
可以很明顯的看出Lambda一方面可以簡省很多代碼,最重要的一點(diǎn)是Lambda表達(dá)式可以避免在抽象類或接口中編寫明確的函數(shù)聲明,進(jìn)而也避免了類的實(shí)現(xiàn)部分(省去了OnClickListener接口這一環(huán)節(jié))
Lambda表達(dá)式語法:
1. lambda 表達(dá)式總是被大括號括著;
2. 其參數(shù)(如果有的話)在 -> 之前聲明(參數(shù)類型可以省略);
3. 函數(shù)體(如果存在的話)在 -> 后面
具體的寫法可以有以下兩種寫法:
// 第一種 val sum1 = {x: Int, j: Int -> x + j} // 第二種 val sum2: (x: Int, j: Int) -> Int = {a, b -> a + b }
分析一下上述兩種表達(dá)式:
第一種比較好理解,首先 ‘=' 左邊聲明了一個變量sum1,'=' 右邊是一個Labmda表達(dá)式,然后將其賦值給sum1
第二種稍微復(fù)雜一點(diǎn),主要是復(fù)雜在左邊的sum2: 后面的這一坨代表什么意思。 首先熟悉Kotlin語言的童鞋應(yīng)該都知道Kotlin函數(shù)參數(shù)是使用 Pascal 表示法定義(name: type), 因此sum2: 后面的這一坨代表的是一種類型type,那具體代表的是什么類型呢? 在Kotlin中一切皆對象,包括函數(shù)也是對象,既然是對象, 同Integer, String等對象一樣,一個函數(shù)也有自己的類型type
(x: Int, j: Int) -> Int這種表述方式就是表達(dá)函數(shù)的類型,它表示的是一個需要傳入兩個Int類型參數(shù),并返回Int類型的函數(shù)。 那么如果想表達(dá)一個無參并返回String類型的函數(shù)該如何表達(dá)呢? 答案見1樓
Lambda傳遞使用
在我們需要使用這兩個Lambda表達(dá)式的時候可以直接將sum1、sum2傳遞給一個高階函數(shù)(稍后講解),或者也可以直接將=之后的表達(dá)式傳遞給高階函數(shù), 具體如下所示:
val view = findViewById(R.id.image) view.setOnClickListener { v -> imageClicked(v) }
接下來我們來看一下,上述的 view.setOnClickListener { v -> imageClicked(v) }是如何一步一步演化而來。在這之前我們需要先了解一下什么是高階函數(shù)
高階函數(shù)是什么
以函數(shù)作為參數(shù)或返回函數(shù)的函數(shù)被稱為高階函數(shù)
定義一個高階函數(shù)
知道了什么是高階函數(shù)之后,我們可以使用一段偽代碼來演示如何定義一個高階函數(shù),如下所示:
fun 高階函數(shù)名(參數(shù)函數(shù)名:參數(shù)函數(shù)類型):高階函數(shù)返回類型{
高階函數(shù)體
...
}
注意:我們姑且將傳入當(dāng)做參數(shù)的函數(shù)起名為參數(shù)函數(shù)
寫一個具體的實(shí)現(xiàn)如下:
fun highOrderFunc(arg1: Int, arg2: Int, paramFunc: (a: Int, b: Int) -> Boolean): Int { return if (paramFunc(arg1, arg2)) { arg1 } else { arg2 } }
上面具體實(shí)例中,我們定義了一個名為highOrderFunc的高階函數(shù),并且傳入了3個參數(shù),前兩個參數(shù)是Int類型, 最后一個參數(shù)是一個函數(shù),并且函數(shù)類型是傳入兩個Int參數(shù)并返回Boolean類型值。最后這個高階函數(shù)自己的返回類型是Int值
使用高階函數(shù)
定義好了一個高階函數(shù)之后,我們就可以將一個Lambda傳遞給這個高階函數(shù),完整實(shí)例如下所示:
fun highOrderFunc(arg1: Int, arg2: Int, paramFunc: (a: Int, b: Int) -> Boolean): Int { return if (paramFunc(arg1, arg2)) { arg1 } else { arg2 } } fun main(args: Array<String>) { val sum1 = {x: Int, j: Int -> x + j} val sum2: (x: Int, j: Int) -> Int = {a, b -> a + b } val max = {x: Int, y: Int -> x > y} println(sum1) println(sum2) println(sum(10, 20)) val biggerNum = highOrderFunc(60, 80, max) println("biggerNum is $biggerNum") }
可以看到,除了sum1和sum2之外,重新定義了一個Lambda函數(shù)val max = {x: Int, y: Int -> x > y}, 并且將此Lambda傳遞給了之前定義的高階函數(shù)highOrderFunc。 這樣綜合起來所表達(dá)的意思就是在傳入的兩個參數(shù)中找出較大的那一個。
最終打印結(jié)果如下:
Function2<java.lang.Integer, java.lang.Integer, java.lang.Integer>
Function2<java.lang.Integer, java.lang.Integer, java.lang.Integer>
30
biggerNum is 80
注意:println(sum1)和println(sum2)打印出來的結(jié)果都是Function2, 這是Kotlin的一個對象,代表的是一個函數(shù)類型
分析
在理解了高階函數(shù)的定義以及使用之后,我們回過頭來理解一下 view.setOnClickListener { v -> imageClicked(v) }這個表達(dá)式是如何一步一步演化而來。
首先我們可以寫一個完整的Lambda,如下所示:
val imageClick: (v: View) -> Unit = {v -> viewClicked(v) }
聲明一個函數(shù)變量imageClick,并指向一個Lambda函數(shù){v -> viewClicked(v) }。 在Lambda函數(shù)體中,調(diào)用了viewClicked(v: View?)方法。然后就可以調(diào)用此方法,完整代碼如下:
class Main2Activity : AppCompatActivity() { override fun onCreate(savedInstanceState: Bundle?) { super.onCreate(savedInstanceState) setContentView(R.layout.activity_main2) // 聲明函數(shù)變量 val imageClick: (v: View) -> Unit = {v -> viewClicked(v) } // 聲明并初始化View對象 val view = View(this) // 調(diào)用View的setOnClickListener方法,設(shè)置點(diǎn)擊監(jiān)聽器,并將imageClick傳進(jìn)去, // 最終點(diǎn)擊ImageView時,會調(diào)用viewClicked方法 view.setOnClickListener(imageClick) } private fun viewClicked(view: View?) { } }
Lambda表達(dá)式也可以傳遞給一個高階函數(shù)當(dāng)做參數(shù),因此上述代碼中
view.setOnClickListener(imageClick),
=>
view.setOnClickListener({v -> viewClicked(v) })
在 Kotlin 中有一個約定,如果函數(shù)的最后一個參數(shù)是一個函數(shù),并且你傳遞一個 lambda 表達(dá)式作為相應(yīng)的參數(shù),你可以在圓括號之外指定它
因此可以實(shí)現(xiàn)如下
view.setOnClickListener({v -> viewClicked(v) })
=>
view.setOnClickListener() {v -> viewClicked(v) }
在 Kotlin中還有另外一個約定,如果一個函數(shù)的參數(shù)只有一個,并且參數(shù)也是一個函數(shù),那么可以省略圓括號
view.setOnClickListener() {v -> viewClicked(v) }
=>
view.setOnClickListener{v -> viewClicked(v) }
總結(jié):
Lambda和高階函數(shù)理解起來有點(diǎn)繞,需要大量的練習(xí)和實(shí)驗(yàn)才能慢慢的理解(一些復(fù)雜的代碼寫的多了 習(xí)慣了之后自然而然的就沒有為什么要這樣寫了 哈哈)
文章一開始我們說了使用Lambda可以省去接口定義和實(shí)現(xiàn)這一環(huán)節(jié),但是是有條件的,此接口必須只有一個抽象方法需要實(shí)現(xiàn),才可以使用Lambda替代(比如OnClickListener、OnItemClickListener)。如果多于1個抽象方法,則不能使用Lambda進(jìn)行替代(比如OnItemSelectedListener)。
具體看如下代碼:
val listView = findViewById(R.id.listView) as ListView listView.onItemClickListener = AdapterView.OnItemClickListener { adapterView, view, i, l -> } listView.onItemSelectedListener = object : AdapterView.OnItemSelectedListener { override fun onItemSelected(adapterView: AdapterView<*>, view: View, i: Int, l: Long) { } override fun onNothingSelected(adapterView: AdapterView<*>) { } }
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Android控件之ProgressBar用法實(shí)例分析
這篇文章主要介紹了Android控件之ProgressBar用法,以一個完整實(shí)例形式較為詳細(xì)的分析了ProgressBar控件操作進(jìn)度顯示的使用技巧,具有一定參考借鑒價值,需要的朋友可以參考下2015-09-09Android實(shí)現(xiàn)狀態(tài)欄(statusbar)漸變效果的示例
本篇文章主要介紹了Android實(shí)現(xiàn)狀態(tài)欄(statusbar)漸變效果的示例,具有一定的參考價值,感興趣的小伙伴們可以參考一下2017-09-09Flutter實(shí)現(xiàn)滑動塊驗(yàn)證碼功能
這篇文章主要為大家詳細(xì)介紹了Flutter實(shí)現(xiàn)滑動塊驗(yàn)證碼功能,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下2022-03-03ReactiveCocoa代碼實(shí)踐之-RAC網(wǎng)絡(luò)請求重構(gòu)
這篇文章主要介紹了ReactiveCocoa代碼實(shí)踐之-RAC網(wǎng)絡(luò)請求重構(gòu) 的相關(guān)資料,需要的朋友可以參考下2016-04-04android開機(jī)自啟動原理與實(shí)現(xiàn)案例(附源碼)
完成一下步驟后,啟動一次程序,完成注冊。等下次手機(jī)開機(jī)時,該軟件即會自動啟動,具體實(shí)現(xiàn)如下,感興趣的朋友可以參考下哈2013-06-06Android編程之匿名內(nèi)部類與回調(diào)函數(shù)用法分析
這篇文章主要介紹了Android編程之匿名內(nèi)部類與回調(diào)函數(shù)用法,結(jié)合實(shí)例形式分析了Android編程中所涉及的java匿名內(nèi)部類與回調(diào)函數(shù)的概念、定義、使用方法與相關(guān)注意事項(xiàng),需要的朋友可以參考下2016-10-10Android高手進(jìn)階教程(二十六)之---Android超仿Path菜單的功能實(shí)現(xiàn)!
本篇文章主要主要介紹了Android超仿Path菜單的功能實(shí)現(xiàn),現(xiàn)在分享給大家,也給大家做個參考。感興趣的可以了解一下。2016-11-11