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

Kotlin 內聯(lián)函數(shù)詳解及實例

 更新時間:2017年06月05日 10:26:15   作者:行云間  
這篇文章主要介紹了Kotlin 內聯(lián)函數(shù)詳解及實例的相關資料,需要的朋友可以參考下

Kotlin 內聯(lián)函數(shù)詳解及實例

概述

在說內聯(lián)函數(shù)之前,先說說函數(shù)的調用過程。

調用某個函數(shù)實際上將程序執(zhí)行順序轉移到該函數(shù)所存放在內存中某個地址,將函數(shù)的程序內容執(zhí)行完后,再返回到轉去執(zhí)行該函數(shù)前的地方。這種轉移操作要求在轉去前要保護現(xiàn)場并記憶執(zhí)行的地址,轉回后先要恢復現(xiàn)場,并按原來保存地址繼續(xù)執(zhí)行。也就是通常說的壓棧和出棧。因此,函數(shù)調用要有一定的時間和空間方面的開銷。那么對于那些函數(shù)體代碼不是很大,又頻繁調用的函數(shù)來說,這個時間和空間的消耗會很大。

那怎么解決這個性能消耗問題呢,這個時候需要引入內聯(lián)函數(shù)了。內聯(lián)函數(shù)就是在程序編譯時,編譯器將程序中出現(xiàn)的內聯(lián)函數(shù)的調用表達式用內聯(lián)函數(shù)的函數(shù)體來直接進行替換。顯然,這樣就不會產生轉去轉回的問題,但是由于在編譯時將函數(shù)體中的代碼被替代到程序中,因此會增加目標程序代碼量,進而增加空間開銷,而在時間代銷上不象函數(shù)調用時那么大,可見它是以目標代碼的增加為代價來換取時間的節(jié)省。

inline

在Kotlin中,使用inline修飾符標記內聯(lián)函數(shù),既會影響到函數(shù)本身, 也影響到傳遞給它的Lambda表達式,這兩者都會被內聯(lián)到調用處。

例如:

inline fun lock<T>(lock: Lock, body: () -> T): T {
// ...
}

編譯器可以直接產生下面的代碼, 而不必為參數(shù)創(chuàng)建函數(shù)對象, 然后再調用這個參數(shù)指向的函數(shù):

l.lock()
try {
  foo()
}
finally {
  l.unlock()
}

noinline

如果一個內聯(lián)函數(shù)的參數(shù)中有多個 Lambda 表達式, 而你只希望內聯(lián)其中的一部分, 你可以對函數(shù)的一部分參數(shù)添加 noinline 標記:

inline fun foo(inlined: () -> Unit, noinline notInlined: () -> Unit) {
  // ...
}

可內聯(lián)的 Lambda 表達式只能在內聯(lián)函數(shù)內部調用, 或者再作為可內聯(lián)的參數(shù)傳遞給其他函數(shù), 但noinline 的 Lambda 表達式可以按照我們喜歡的方式任意使用: 可以保存在域內, 也可以當作參數(shù)傳遞, 等等。

非局部返回(Non-local return)

在Kotlin中, 使用無限定符的通常的return語句, 只能用來退出一個有名稱的函數(shù), 或匿名函數(shù). 這就意味著, 要退出一個Lambda表達式, 我們必須使用一個 標簽, 無標簽的 return 在 Lambda 表達式內是禁止使用的, 因為 Lambda 表達式不允許強制包含它的函數(shù)返回:

fun foo() {
  ordinaryFunction {
    return // 錯誤: 這里不允許讓 `foo` 函數(shù)返回
  }
}

如果 Lambda 表達式被傳遞去的函數(shù)是內聯(lián)函數(shù), 那么 return 語句也可以內聯(lián), 因此 return 是允許的。

fun foo() {
  inlineFunction {
    return // OK: 這里的 Lambda 表達式是內聯(lián)的
  }
}

注:

有些內聯(lián)函數(shù)可能并不在自己的函數(shù)體內直接調用傳遞給它的 Lambda 表達式參數(shù), 而是通過另一個執(zhí)行環(huán)境來調用, 比如通過一個局部對象, 或者一個嵌套函數(shù). 這種情況下, 在 Lambda 表達式內, 非局部的控制流同樣是禁止的. 為了標識這一點, Lambda 表達式參數(shù)需要添加 crossinline修飾符。

inline fun f(crossinline body: () -> Unit) {
  val f = object: Runnable {
    override fun run() = body()
  }
  // ...
}


感謝閱讀,希望能幫助到大家,謝謝大家對本站的支持!

相關文章

最新評論