Kotlin 內(nèi)聯(lián)函數(shù)詳解及實例
Kotlin 內(nèi)聯(lián)函數(shù)詳解及實例
概述
在說內(nèi)聯(lián)函數(shù)之前,先說說函數(shù)的調(diào)用過程。
調(diào)用某個函數(shù)實際上將程序執(zhí)行順序轉(zhuǎn)移到該函數(shù)所存放在內(nèi)存中某個地址,將函數(shù)的程序內(nèi)容執(zhí)行完后,再返回到轉(zhuǎn)去執(zhí)行該函數(shù)前的地方。這種轉(zhuǎn)移操作要求在轉(zhuǎn)去前要保護現(xiàn)場并記憶執(zhí)行的地址,轉(zhuǎn)回后先要恢復(fù)現(xiàn)場,并按原來保存地址繼續(xù)執(zhí)行。也就是通常說的壓棧和出棧。因此,函數(shù)調(diào)用要有一定的時間和空間方面的開銷。那么對于那些函數(shù)體代碼不是很大,又頻繁調(diào)用的函數(shù)來說,這個時間和空間的消耗會很大。
那怎么解決這個性能消耗問題呢,這個時候需要引入內(nèi)聯(lián)函數(shù)了。內(nèi)聯(lián)函數(shù)就是在程序編譯時,編譯器將程序中出現(xiàn)的內(nèi)聯(lián)函數(shù)的調(diào)用表達式用內(nèi)聯(lián)函數(shù)的函數(shù)體來直接進行替換。顯然,這樣就不會產(chǎn)生轉(zhuǎn)去轉(zhuǎn)回的問題,但是由于在編譯時將函數(shù)體中的代碼被替代到程序中,因此會增加目標(biāo)程序代碼量,進而增加空間開銷,而在時間代銷上不象函數(shù)調(diào)用時那么大,可見它是以目標(biāo)代碼的增加為代價來換取時間的節(jié)省。
inline
在Kotlin中,使用inline修飾符標(biāo)記內(nèi)聯(lián)函數(shù),既會影響到函數(shù)本身, 也影響到傳遞給它的Lambda表達式,這兩者都會被內(nèi)聯(lián)到調(diào)用處。
例如:
inline fun lock<T>(lock: Lock, body: () -> T): T { // ... }
編譯器可以直接產(chǎn)生下面的代碼, 而不必為參數(shù)創(chuàng)建函數(shù)對象, 然后再調(diào)用這個參數(shù)指向的函數(shù):
l.lock() try { foo() } finally { l.unlock() }
noinline
如果一個內(nèi)聯(lián)函數(shù)的參數(shù)中有多個 Lambda 表達式, 而你只希望內(nèi)聯(lián)其中的一部分, 你可以對函數(shù)的一部分參數(shù)添加 noinline 標(biāo)記:
inline fun foo(inlined: () -> Unit, noinline notInlined: () -> Unit) { // ... }
可內(nèi)聯(lián)的 Lambda 表達式只能在內(nèi)聯(lián)函數(shù)內(nèi)部調(diào)用, 或者再作為可內(nèi)聯(lián)的參數(shù)傳遞給其他函數(shù), 但noinline 的 Lambda 表達式可以按照我們喜歡的方式任意使用: 可以保存在域內(nèi), 也可以當(dāng)作參數(shù)傳遞, 等等。
非局部返回(Non-local return)
在Kotlin中, 使用無限定符的通常的return語句, 只能用來退出一個有名稱的函數(shù), 或匿名函數(shù). 這就意味著, 要退出一個Lambda表達式, 我們必須使用一個 標(biāo)簽, 無標(biāo)簽的 return 在 Lambda 表達式內(nèi)是禁止使用的, 因為 Lambda 表達式不允許強制包含它的函數(shù)返回:
fun foo() { ordinaryFunction { return // 錯誤: 這里不允許讓 `foo` 函數(shù)返回 } }
如果 Lambda 表達式被傳遞去的函數(shù)是內(nèi)聯(lián)函數(shù), 那么 return 語句也可以內(nèi)聯(lián), 因此 return 是允許的。
fun foo() { inlineFunction { return // OK: 這里的 Lambda 表達式是內(nèi)聯(lián)的 } }
注:
有些內(nèi)聯(lián)函數(shù)可能并不在自己的函數(shù)體內(nèi)直接調(diào)用傳遞給它的 Lambda 表達式參數(shù), 而是通過另一個執(zhí)行環(huán)境來調(diào)用, 比如通過一個局部對象, 或者一個嵌套函數(shù). 這種情況下, 在 Lambda 表達式內(nèi), 非局部的控制流同樣是禁止的. 為了標(biāo)識這一點, Lambda 表達式參數(shù)需要添加 crossinline修飾符。
inline fun f(crossinline body: () -> Unit) { val f = object: Runnable { override fun run() = body() } // ... }
感謝閱讀,希望能幫助到大家,謝謝大家對本站的支持!
相關(guān)文章
Java設(shè)計模塊系列之書店管理系統(tǒng)單機版(二)
這篇文章主要為大家詳細介紹了Java單機版的書店管理系統(tǒng)設(shè)計模塊和思想第二章,感興趣的小伙伴們可以參考一下2016-08-08Java安全框架——Shiro的使用詳解(附springboot整合Shiro的demo)
這篇文章主要介紹了Java安全框架——Shiro的使用詳解,幫助大家更好的理解和學(xué)習(xí)使用Shiro,感興趣的朋友可以了解下2021-04-04SpringBoot如何通過@Profile注解配置多環(huán)境
在Spring中,可以使用配置文件的方式來指定不同環(huán)境下所需要的配置信息,本文給大家介紹SpringBoot如何通過@Profile注解配置多環(huán)境,感興趣的朋友跟隨小編一起看看吧2023-06-06Java中FTPClient上傳中文目錄、中文文件名亂碼問題解決方法
這篇文章主要介紹了Java中FTPClient上傳中文目錄、中文文件名亂碼問題解決方法,本文使用apache-commons-net工具包時遇到這個問題,解決方法很簡單,需要的朋友可以參考下2015-05-05Java8時間轉(zhuǎn)換(LocalDateTime)代碼實例
這篇文章主要介紹了java8時間轉(zhuǎn)換(LocalDateTime)代碼實例,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下2019-11-11