麻豆小视频在线观看_中文黄色一级片_久久久成人精品_成片免费观看视频大全_午夜精品久久久久久久99热浪潮_成人一区二区三区四区

首頁 > 開發 > 綜合 > 正文

玩轉Kotlin 徹底弄懂Lambda和高階函數

2024-07-21 23:03:39
字體:
來源:轉載
供稿:網友

Lambda是什么

簡單來講,Lambda是一種函數的表示方式(言外之意也就是說一個Lambda表達式等于一個函數)。更確切的說:Lambda是一個未聲明的函數,會以表達式的形式傳遞

為什么要用Lambda

設想一下,在Android中實現一個View的點擊事件,可以使用如下實現:

View view = findViewById(R.id.textView);view.setOnClickListener(new View.OnClickListener() {    @Override      public void onClick(View view) {        viewClicked(view);      }   });

而如果在Kotlin中使用Lambda,則實現可以簡單如下:

val view = findViewById(R.id.image)view.setOnClickListener { v -> viewClicked(v) }

可以很明顯的看出Lambda一方面可以簡省很多代碼,最重要的一點是Lambda表達式可以避免在抽象類或接口中編寫明確的函數聲明,進而也避免了類的實現部分(省去了OnClickListener接口這一環節)

Lambda表達式語法:

1. lambda 表達式總是被大括號括著;
2. 其參數(如果有的話)在 -> 之前聲明(參數類型可以省略);
3. 函數體(如果存在的話)在 -> 后面

具體的寫法可以有以下兩種寫法:

// 第一種val sum1 = {x: Int, j: Int -> x + j}// 第二種val sum2: (x: Int, j: Int) -> Int = {a, b -> a + b }

分析一下上述兩種表達式:

第一種比較好理解,首先 ‘=' 左邊聲明了一個變量sum1,'=' 右邊是一個Labmda表達式,然后將其賦值給sum1
第二種稍微復雜一點,主要是復雜在左邊的sum2: 后面的這一坨代表什么意思。 首先熟悉Kotlin語言的童鞋應該都知道Kotlin函數參數是使用 Pascal 表示法定義(name: type), 因此sum2: 后面的這一坨代表的是一種類型type,那具體代表的是什么類型呢? 在Kotlin中一切皆對象,包括函數也是對象,既然是對象, 同Integer, String等對象一樣,一個函數也有自己的類型type
(x: Int, j: Int) -> Int這種表述方式就是表達函數的類型,它表示的是一個需要傳入兩個Int類型參數,并返回Int類型的函數。 那么如果想表達一個無參并返回String類型的函數該如何表達呢? 答案見1樓

Lambda傳遞使用

在我們需要使用這兩個Lambda表達式的時候可以直接將sum1、sum2傳遞給一個高階函數(稍后講解),或者也可以直接將=之后的表達式傳遞給高階函數, 具體如下所示:

val view = findViewById(R.id.image)view.setOnClickListener { v -> imageClicked(v) }

接下來我們來看一下,上述的 view.setOnClickListener { v -> imageClicked(v) }是如何一步一步演化而來。在這之前我們需要先了解一下什么是高階函數

高階函數是什么

以函數作為參數或返回函數的函數被稱為高階函數

定義一個高階函數

知道了什么是高階函數之后,我們可以使用一段偽代碼來演示如何定義一個高階函數,如下所示:

fun 高階函數名(參數函數名:參數函數類型):高階函數返回類型{
    高階函數體
    ...
}

注意:我們姑且將傳入當做參數的函數起名為參數函數

寫一個具體的實現如下:

fun highOrderFunc(arg1: Int, arg2: Int, paramFunc: (a: Int, b: Int) -> Boolean): Int {  return if (paramFunc(arg1, arg2)) {    arg1  } else {    arg2  }}

上面具體實例中,我們定義了一個名為highOrderFunc的高階函數,并且傳入了3個參數,前兩個參數是Int類型, 最后一個參數是一個函數,并且函數類型是傳入兩個Int參數并返回Boolean類型值。最后這個高階函數自己的返回類型是Int值

使用高階函數

定義好了一個高階函數之后,我們就可以將一個Lambda傳遞給這個高階函數,完整實例如下所示:

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函數val max = {x: Int, y: Int -> x > y}, 并且將此Lambda傳遞給了之前定義的高階函數highOrderFunc。 這樣綜合起來所表達的意思就是在傳入的兩個參數中找出較大的那一個。

最終打印結果如下:

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)打印出來的結果都是Function2, 這是Kotlin的一個對象,代表的是一個函數類型

分析

在理解了高階函數的定義以及使用之后,我們回過頭來理解一下 view.setOnClickListener { v -> imageClicked(v) }這個表達式是如何一步一步演化而來。
首先我們可以寫一個完整的Lambda,如下所示:

val imageClick: (v: View) -> Unit = {v -> viewClicked(v) }

聲明一個函數變量imageClick,并指向一個Lambda函數{v -> viewClicked(v) }。 在Lambda函數體中,調用了viewClicked(v: View?)方法。然后就可以調用此方法,完整代碼如下:

class Main2Activity : AppCompatActivity() {  override fun onCreate(savedInstanceState: Bundle?) {    super.onCreate(savedInstanceState)    setContentView(R.layout.activity_main2)    // 聲明函數變量    val imageClick: (v: View) -> Unit = {v -> viewClicked(v) }    // 聲明并初始化View對象    val view = View(this)    // 調用View的setOnClickListener方法,設置點擊監聽器,并將imageClick傳進去,    // 最終點擊ImageView時,會調用viewClicked方法    view.setOnClickListener(imageClick)  }  private fun viewClicked(view: View?) {  }}

Lambda表達式也可以傳遞給一個高階函數當做參數,因此上述代碼中
view.setOnClickListener(imageClick),
=>
view.setOnClickListener({v -> viewClicked(v) })

在 Kotlin 中有一個約定,如果函數的最后一個參數是一個函數,并且你傳遞一個 lambda 表達式作為相應的參數,你可以在圓括號之外指定它
因此可以實現如下
view.setOnClickListener({v -> viewClicked(v) })
=>
view.setOnClickListener() {v -> viewClicked(v) }

在 Kotlin中還有另外一個約定,如果一個函數的參數只有一個,并且參數也是一個函數,那么可以省略圓括號
view.setOnClickListener() {v -> viewClicked(v) }
=>
view.setOnClickListener{v -> viewClicked(v) }

總結:

Lambda和高階函數理解起來有點繞,需要大量的練習和實驗才能慢慢的理解(一些復雜的代碼寫的多了 習慣了之后自然而然的就沒有為什么要這樣寫了 哈哈)

文章一開始我們說了使用Lambda可以省去接口定義和實現這一環節,但是是有條件的,此接口必須只有一個抽象方法需要實現,才可以使用Lambda替代(比如OnClickListener、OnItemClickListener)。如果多于1個抽象方法,則不能使用Lambda進行替代(比如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<*>) {      }    }

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持VeVb武林網。


注:相關教程知識閱讀請移步到kotlin教程頻道。
發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
主站蜘蛛池模板: 欧美一级做一级爱a做片性 久久久资源网 | 毛片免费观看视频 | 9999免费视频| 久久国产精品二国产精品中国洋人 | 成人一级免费视频 | wwwxxx免费视频 | 激情久久精品 | 国产精品视频一区二区三区四 | 羞羞视频免费网站 | 在线观看va | 日日鲁夜夜视频热线播放 | 污污黄 | 久久精品毛片 | 久久久久久久久久久一区 | 国产精品久久久久久久久久了 | 国产一级毛片国语版 | av免费在线播放网址 | 久久经典国产视频 | 国产午夜亚洲精品理论片大丰影院 | 黄色网战入口 | 狠狠一区| 国产在线精品一区二区三区不卡 | 欧美一级高潮片免费的 | 男男啪羞羞视频网站 | 蜜桃av网 | 久久精品视频12 | 国产a级片电影 | 成人网在线观看 | 亚洲精品午夜在线 | 久久国产免费视频 | 国产精品一区二区三区在线看 | 国产午夜精品在线 | 狠狠操电影 | 久久久成人一区二区免费影院 | 成人污在线 | 国产精品入口夜色视频大尺度 | 亚洲影视在线 | 嫩草影院在线观看网站成人 | 92看片淫黄大片欧美看国产片 | 亚洲一区二区三区四区精品 | 99国产精品白浆在线观看免费 |