Kotlin中關(guān)于內(nèi)聯(lián)函數(shù)的一些理解分享
前言
看了很多博客,才明白了內(nèi)聯(lián)的含義,其實最根本的就是將寫在別處的代碼拷貝到你現(xiàn)在執(zhí)行的方法中,相當于在一個方法中執(zhí)行,java的方法執(zhí)行是需要壓棧出棧的對吧,如果是兩三個方法那就是兩三次的壓棧出棧,為了節(jié)省這個操作,提高一定的效率,kotlin就出了這么個函數(shù)。但又想想,如果是個超級大的函數(shù),考來考去的也是很麻煩啊,所以這東西需要自己權(quán)衡吧,遵守單一職責,降低代碼圈發(fā)雜度才是根本。
內(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<String>) { doSomething { pringln("Hello World") } }
上面的調(diào)用會被編譯成:
fun main(args: Array<String>) { println("Before doSomething...") println("Hello World") println("After doSomething...") }
從上面編譯的結(jié)果可以看出,無論doSomething函數(shù)還是action參數(shù)都被內(nèi)聯(lián)了,很棒,那讓我們換一種調(diào)用方式:
fun main(args: Array<String>) { val action:() -> Unit = { println("Hello World") } doSomething(action) }
上面的調(diào)用會被編譯成:
fun main(args: Array<String>) { 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é):
- 當lambda表達式以參數(shù)的形式直接傳遞給內(nèi)聯(lián)函數(shù),那么lambda表達式的代碼會被直接替換到最終生成的代碼中。
- 當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ù)的全部理解,通過掌握該特性的運行機制,相信大家可以做到在正確的時機使用該特性,而非濫用或因恐懼棄而不用。
總結(jié)
以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,如果有疑問大家可以留言交流,謝謝大家對腳本之家的支持。
相關(guān)文章
Android AAPT(Android Asset Packaging Too
AAPT - Android Asset Packaging Tool作用AAPT基本命令A(yù)APT編譯資源源碼解析AAPT打包和系統(tǒng)不一致的資源2024-04-04Android自定義view實現(xiàn)多色進度條GradientProgressView的繪制
我們常使用shape實現(xiàn)漸變色,但是shape的極限卻只有三色,如果有超過三種顏色的View的要求,那么我們就不得不去自定義View來實現(xiàn)這個需求,所以下面我們就來看看如何自定義view實現(xiàn)多色進度條的繪制吧2023-08-08如何通過Android Logcat插件分析firebase崩潰問題
android crash Crash(應(yīng)用崩潰)是由于代碼異常而導(dǎo)致App非正常退出,導(dǎo)致應(yīng)用程序無法繼續(xù)使用,所有工作都停止的現(xiàn)象,本文重點介紹如何通過Android Logcat插件分析firebase崩潰問題,感興趣的朋友一起看看吧2024-01-01