欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

擁抱kotlin之如何習慣使用kotlin高階函數(shù)

 更新時間:2019年12月29日 09:46:01   作者:Dynamic  
這篇文章主要給大家介紹了關于擁抱kotlin之如何習慣使用kotlin高階函數(shù)的相關資料,文中通過示例代碼介紹的非常詳細,對大家學習或者使用kotlin具有一定的參考學習價值,需要的朋友們下面來一起學習學習吧

前言

kotlin提供了高階函數(shù)這個概念,可以在一些場景提高編碼效率

一、什么是高階函數(shù)

通俗的說和數(shù)學里面的高階函數(shù)概念類似,也就是函數(shù)里面的參數(shù)可以是函數(shù)。當然返回值也可以是函數(shù)。

二、kotlin高階函數(shù)使用場景分析

1.先看看平時使用比較多的內置高階函數(shù)

用kotlin寫view的onClickListener

 tV.setOnClickListener {
   //doSomeThing
  }

里面的lamba表達式就是一個函數(shù)

不太形象?再看看集合里面的filter、map

listOf(1, 2, 3)
   .filter { it > 2 }
   .map { it + 5 }

/**
 * Returns a list containing only elements matching the given [predicate].
 */
public inline fun <T> Iterable<T>.filter(predicate: (T) -> Boolean): List<T> {
 return filterTo(ArrayList<T>(), predicate)
}

filter、map的參數(shù)都是一個lambda函數(shù)

2.高階函數(shù)有什么用

就拿filter函數(shù)來說,比如實現(xiàn)一個過濾的邏輯,判斷是符合的

若classA 和classB都需要調用這個函數(shù),那么函數(shù)就需要兼容這兩種情況

fun filter(): Boolean {
  if (classA) {
   return true
  } else if (classB) {
   return false
  }
  return false
 }

if else無可厚非,但是如果后面有classC classD...都需要考慮呢,這顯然違背了開閉原則。那么自然是要面向抽象而不是具體,當然就是抽象類或者接口。

若用java的方式去實現(xiàn),會變成這樣

interface IJudge {
  fun canFilter(): Boolean
 }

 class ClassA : IJudge {
  override fun canFilter(): Boolean {
   return true
  }
 }

 class ClassB : IJudge {
  override fun canFilter(): Boolean {
   return false
  }
 }

 fun filter(a:Int,b:Int,jugde: IJudge): Boolean {
  //加一些邏輯
  return jugde.canFilter()
 }

這個是硬傷,面向抽象就得加這么接口,然后多寫一些代碼。

若用高階函數(shù)實現(xiàn)

  fun filter(a: Int, b: Int, canFilter: (a:Int,b:Int) -> Boolean): Boolean {
  //加一些邏輯
  return canFilter(a,b)
 }
  //調用方1
  filter(1, 2) { a: Int, b: Int ->
   a * b > 10
  }
  //調用方2
  filter(1, 2) { a: Int, b: Int ->
   a + b < 5
  }

這樣就省了個接口,后面分析實際是編譯器幫忙處理,其實還是生成了接口

三、kotlin高階函數(shù)的實現(xiàn)

來看看kotlin編譯器是怎么實現(xiàn)的吧

首先把上面那段kotlin代碼反編譯成java

kt:
  fun filter(a: Int, b: Int, canFilter: (a:Int,b:Int) -> Boolean): Boolean {
    //加一些邏輯
    return canFilter(a,b)
  }
java:
 public final boolean filter(int a, int b, @NotNull Function2 canFilter) {
   Intrinsics.checkParameterIsNotNull(canFilter, "canFilter");
   canFilter.invoke(a, b);
   return (Boolean)canFilter.invoke(a, b);
  }

實際上是kt內置的 Functions.kt

這里由于我傳的是2個參數(shù)的lambda函數(shù),所以調用的是Function2

那么從這里能得來上面結論:

a.高階函數(shù)所謂的可以省略接口,其實只能省略只有一個方法的接口,因為function函數(shù)只有一個方法

b.上邊的fliter函數(shù)除了canFIlter(a,b)還可以使用canFilter.invoke(a,b)調用。這個在需要對函數(shù)判空的時候很有用。比如替換只有一個方法的接口回調可以callback?.invoke(a,b,c) , 因為callbck?(a,b,c)是不能編譯通過的。

c.雖然Functions.kt文件方法數(shù)是有限的,感覺意味著lambda參數(shù)是有限的,最多22個參數(shù),超過會編譯失敗。但是當真的超過時,會調用另外一個FunctionN.kt

operator fun invoke(vararg args: Any?): R

不過如果誰寫的函數(shù),直接傳參20多個還不封成對象或者builder,怕是腿都要被打斷.......

四、關于高階函數(shù)替換接口的討論

上面已經討論了,當接口只有一個方法時,確實可以用高階函數(shù)代替,省略一個接口。

但是當接口有多個方法時,顯然不能直接替換。雖然也可以把幾個函數(shù)包裝在一起使用,但是還是感覺多此一舉。
多人并行開發(fā)的時候,比如一個人負責寫一個負責ui,一個負責使用ui處理業(yè)務邏輯。先把接口定好,接口方法文檔寫好,一目了然。這一方面還是接口好很多,當只有簡單的一個方法時,用高階函數(shù)要方便一些。

總結

以上就是這篇文章的全部內容了,希望本文的內容對大家的學習或者工作具有一定的參考學習價值,謝謝大家對腳本之家的支持。

相關文章

最新評論