這篇文章將為大家詳細講解有關(guān)Kotlin中內(nèi)聯(lián)函數(shù)的示例分析,小編覺得挺實用的,因此分享給大家做個參考,希望大家閱讀完這篇文章后可以有所收獲。
開平ssl適用于網(wǎng)站、小程序/APP、API接口等需要進行數(shù)據(jù)傳輸應(yīng)用場景,ssl證書未來市場廣闊!成為創(chuàng)新互聯(lián)公司的ssl證書銷售渠道,可以享受市場價格4-6折優(yōu)惠!如果有意向歡迎電話聯(lián)系或者加微信:028-86922220(備注:SSL證書合作)期待與您的合作!
內(nèi)聯(lián)函數(shù)的理解
inline函數(shù)(內(nèi)聯(lián)函數(shù))從概念上講是編譯器使用函數(shù)實現(xiàn)的真實代碼來替換每一次的函數(shù)調(diào)用,帶來的最直接的好處就是節(jié)省了函數(shù)調(diào)用的開銷,而缺點就是增加了所生成字節(jié)碼的尺寸。基于此,在代碼量不是很大的情況下,我們是否有必要將所有的函數(shù)定義為內(nèi)聯(lián)?讓我們分兩種情況進行說明:
將普通函數(shù)定義為內(nèi)聯(lián):眾所周知,JVM內(nèi)部已經(jīng)實現(xiàn)了內(nèi)聯(lián)優(yōu)化,它會在任何可以通過內(nèi)聯(lián)來提升性能的地方將函數(shù)調(diào)用內(nèi)聯(lián)化,并且相對于手動將普通函數(shù)定義為內(nèi)聯(lián),通過JVM內(nèi)聯(lián)優(yōu)化所生成的字節(jié)碼,每個函數(shù)的實現(xiàn)只會出現(xiàn)一次,這樣在保證減少運行時開銷的同時,也沒有增加字節(jié)碼的尺寸;所以我們可以得出結(jié)論,對于普通函數(shù),我們沒有必要將其聲明為內(nèi)聯(lián)函數(shù),而是交給JVM自行優(yōu)化。
將帶有l(wèi)ambda參數(shù)的函數(shù)定義為內(nèi)聯(lián):是的,這種情況下確實可以提高性能;但在使用的過程中,我們會發(fā)現(xiàn)它是有諸多限制的,讓我們從下面的例子開始展開說明:
inline fun doSomething(action: () -> Unit) { println("Before doSomething...") action() println("After doSomething...") }
假如我們這樣調(diào)用doSomething:
fun main(args: Array) { doSomething { pringln("Hello World") } }
上面的調(diào)用會被編譯成:
fun main(args: Array) { println("Before doSomething...") println("Hello World") println("After doSomething...") }
從上面編譯的結(jié)果可以看出,無論doSomething函數(shù)還是action參數(shù)都被內(nèi)聯(lián)了,很棒,那讓我們換一種調(diào)用方式:
fun main(args: Array) { val action:() -> Unit = { println("Hello World") } doSomething(action) }
上面的調(diào)用會被編譯成:
fun main(args: Array) { println("Before doSomething...") action() println("After doSomething...") }
doSomething函數(shù)被內(nèi)聯(lián),而action參數(shù)沒有被內(nèi)聯(lián),這是因為以函數(shù)型變量的形式傳遞給doSomething的lambda在函數(shù)的調(diào)用點是不可用的,只有等到doSomething被內(nèi)聯(lián)后,該lambda才可以正常使用。
通過上面的例子,我們對lambda表達式何時被內(nèi)聯(lián)做一下簡單的總結(jié):
當(dāng)lambda表達式以參數(shù)的形式直接傳遞給內(nèi)聯(lián)函數(shù),那么lambda表達式的代碼會被直接替換到最終生成的代碼中。
當(dāng)lambda表達式在某個地方被保存起來,然后以變量形式傳遞給內(nèi)聯(lián)函數(shù),那么此時的lambda表達式的代碼將不會被內(nèi)聯(lián)。
上面對lambda的內(nèi)聯(lián)時機進行了討論,消化片刻后讓我們再看最后一個例子:
inline fun doSomething(action: () -> Unit, secretAction: () -> Unit) { action() doSomethingSecret(secretAction) } fun doSomethingSecret(secretAction: () -> Unit) { }
上面的例子是否有問題?是的,編譯器會拋出“Illegal usage of inline-parameter”的錯誤,這是因為Kotlin規(guī)定內(nèi)聯(lián)函數(shù)中的lambda參數(shù)只能被直接調(diào)用或者傳遞給另外一個內(nèi)聯(lián)函數(shù),除此之外不能作為他用;那我們?nèi)绻_實想要將某一個lambda傳遞給一個非內(nèi)聯(lián)函數(shù)怎么辦?我們只需將上述代碼這樣改造即可:
inline fun doSomething(action: () -> Unit, noinline secretAction: () -> Unit) { action() doSomethingSecret(secretAction) } fun doSomethingSecret(secretAction: () -> Unit) { }
很簡單,在不需要內(nèi)聯(lián)的lambda參數(shù)前加上noinline修飾符就可以了。
以上便是我對內(nèi)聯(lián)函數(shù)的全部理解,通過掌握該特性的運行機制,相信大家可以做到在正確的時機使用該特性,而非濫用或因恐懼棄而不用。
關(guān)于“Kotlin中內(nèi)聯(lián)函數(shù)的示例分析”這篇文章就分享到這里了,希望以上內(nèi)容可以對大家有一定的幫助,使各位可以學(xué)到更多知識,如果覺得文章不錯,請把它分享出去讓更多的人看到。