Kotlin-函數和lambda表達式學習筆記

1.Kotlin函數可以使用中綴表示法來調用,當滿足一下條件時:

infix fun Int.add(x:Int):Int{
    return this+x
}

fun printAdd(){
    val value=2 add 3//中綴表示法調用add函數(不用加".")
    print("the value is $value")
}

2.在Kotlin中,調用函數的時候可以使用命名的參數:

fun foo(arg1:String="Hello Kotlin",arg2:Boolean,arg3:Int){
    print("arg1="+arg1+"&arg2="+arg2+"arg3="+arg3)
}

fun printFoo(){
    foo(arg2 = true,arg3 = 7)//命名參數調用foo函數
}

3.Kotlin函數定義可變數量的參數,只要用vararg關鍵字修飾即可:

//可變數量的參數
fun foo4(vararg args:Int){
    for (arg in args){
        print(arg.toString()+",")
    }
}

fun printFoo4(){
    foo4(1,2,3,4,5)//1,2,3,4,5,
}

4.Kotlin函數以數組的內容作為參數,只要用伸展(spread)操作符(在數組前面加 *):

fun foo4(vararg args:Int){
    for (arg in args){
        print(arg.toString()+",")
    }
}

fun printFoo4(){
    val values= intArrayOf(1,2,3,4,5)
    foo4(*values)//使用擴展運算符傳一個數組給可變參數
}

5.高階函數

高階函數可以將一個函數作為參數或返回值:

fun add2(x:Int=0,y:Int=0):Int{
    return x+y
}

fun operate(x:Int=0,y:Int=0,body:(Int,Int)->Int){//body是一個函數類型,傳入兩個Int類型參數,返回一個Int類型參數
    print("this result is "+body(x,y))
}

fun getValue(){
    operate(3,7,::add2)
}

6.Lambda表達式

Lambda表達式通常使用“{ }”包圍,參數是定義在“()”內,實體部分跟在“->”后面;
以上的operate()方法,我們還有一個更簡潔的調用方式,即傳入一個lambda表達式:

operate(3,7,{x,y->x+y})//函數參數傳入一個lambda表達式

當函數參數是最后函數的最后一個參數,并且你傳入一個lambda表達式作為相應的參數,則可以在圓括號之外指定它:

operate(3,7){//函數參數作為函數的最后一個參數,并且傳入的是一個lambda表達式,可以在在圓括號外指定
        x,y->x+y

如果lambda表達式只有一個參數,kotlin可以自己計算出簽名,它允許我們不聲明唯一的參數,并且隱含的為我們聲明其名稱為it:

fun upCase(str:String,body:(String)->String):String{//body是一個函數參數,傳入一個String類型參數,返回一個String類型
    return body(str)
}
fun transform(){
    upCase("HelloKotlin"){//函數字面值只有一個參數,可以省略參數聲明,其名稱是it
        it.toUpperCase()
    }
}

如果lambda表達式是調用的唯一參數,則調用中的圓括號可以完全省略。

7.匿名函數

匿名函數與常規函數一樣,只是省略了函數名稱:

fun String.upper(body:(String)->String):String{
    return body(this)
}

fun transform(){
    "HelloKotlin".upper { it.toUpperCase() }//lambda表達式是調用的唯一參數,則調用的圓括號可以省略
    "HelloKotlin".upper(fun(str:String):String{//將匿名函數作為一個函數參數傳入
        return str.toUpperCase()
    })
}

匿名函數 與 Lambda表示式區別:

  • 匿名函數作為參數,一般定義在“()”中;而Lambda表達式可以定義到調用函數“()”外。
  • 另外區別在“非局部返回(non-local returns)”行為上:非標簽注解的return(返回對應的最內層的函數(即fun)),在匿名函數中,退出該匿名函數;而在Lambda表達中,退出包含該表達式的函數。
fun foo() {
    ints.forEach {
        if (it == 0) return//這個 return 表達式從最直接包圍它的函數即 foo 中返回。
        print(it)
    }
}
fun transform():String{
    "HelloKotlin".upper {
        print(it.toUpperCase())
        return@upper it.toUpperCase()//返回必須加標簽限制
    }
    "HelloKotlin".upper(fun(str:String):String{
        return str.toUpperCase()//從匿名函數返回
    })
}

8.內嵌函數

作為自變量接收的函數令人討厭的是編譯器需要為它們創建類,這將影響性能。但是,這可以用保留字inline容易地解決。

由于Inline函數是在編譯時將它的代碼替換對它的調用,所以在性能方面影響較少。為此,它不需要用額為的對象。

我們能夠轉換doAsync到一個inline函數:

inline fun doAsync(crossinline f: () -> Unit) {
    Thread({ f() }).start()
}

在這個例子中,由于我們從另一個執行內容(另一個Lambda表達式)中調用f(),所以要求crossinline。不用太擔心這些,在需要使用它時,編譯器會提醒你的。

現在,在我們的代碼中,產生異步代碼塊:

doAsync { 
    op1()
    op2()
    op3()
}

這樣在{}內的每件事都將在第二個線程中執行。

參考文章

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。

推薦閱讀更多精彩內容