kotlin知識體系(四) : inline、noinline、crossinline 關鍵字對應編譯后的代碼是怎樣的 ?

kotlin中inline、noinline、crossinline 關鍵字的作用

在 Kotlin 里,inlinenoinlinecrossinline 這幾個關鍵字和高階函數緊密相關,它們能夠對高階函數的行為進行優化和控制。下面為你詳細闡述它們的作用和原理。

inline 關鍵字

inline 關鍵字用于修飾高階函數,其作用是在編譯時將函數調用處替換為函數體本身,以此避免函數調用的開銷,提高代碼的執行效率。

示例代碼
// 定義一個內聯高階函數
inline fun inlineFunction(block: () -> Unit) {block()
}fun main() {inlineFunction {println("This is an inline function call.")}
}
代碼解釋

在上述示例中,inlineFunctioninline 關鍵字修飾。在編譯時,inlineFunction 的調用會被替換為函數體內容,這樣就不會有額外的函數調用開銷。不過,使用 inline 也會使生成的字節碼體積增大,因為函數體被復制到了調用處。

noinline 關鍵字

當高階函數被 inline 修飾時,它的所有函數參數默認也會被內聯。要是你不希望某個函數參數被內聯,就可以使用 noinline 關鍵字。

示例代碼
// 定義一個內聯高階函數,包含一個 noinline 參數
inline fun mixedFunction(inlineBlock: () -> Unit, noinline noInlineBlock: () -> Unit) {inlineBlock()noInlineBlock()
}fun main() {mixedFunction({ println("This is an inline block.") },{ println("This is a non - inline block.") })
}
代碼解釋

在這個例子中,mixedFunction 是內聯函數,inlineBlock 會被內聯,而 noInlineBlock 由于使用了 noinline 關鍵字,不會被內聯。noinline 通常用于需要將函數參數存儲在變量中或者作為其他函數的返回值的情況。

crossinline 關鍵字

在使用 inline 修飾高階函數時,內聯函數參數里不允許有非局部返回(即從外層函數返回)。若需要在 Lambda 表達式中使用 return 語句,但又不想使用 noinline 來避免內聯,就可以使用 crossinline 關鍵字。

示例代碼
// 定義一個內聯高階函數,包含一個 crossinline 參數
inline fun crossInlineFunction(crossinline block: () -> Unit) {val wrapper = {block()}wrapper()
}fun main() {crossInlineFunction {// 這里不能使用 return 進行非局部返回,但可以執行其他操作println("Inside crossinline block.")}
}
代碼解釋

在這個例子中,crossInlineFunction 是內聯函數,block 參數使用了 crossinline 關鍵字。在 block 中不能使用非局部返回,但可以正常執行其他操作。這樣既能保證參數被內聯,又能在一定程度上控制返回行為。

綜上所述,inlinenoinlinecrossinline 關鍵字在 Kotlin 中用于控制高階函數及其參數的內聯行為,有助于優化代碼性能和控制函數返回邏輯。

kotlin中inline、noinline、crossinline 關鍵字對應編譯后的代碼是怎樣的 ?

下面通過具體示例,詳細分析 Kotlin 中 inlinenoinlinecrossinline 關鍵字在編譯后代碼的表現。

1. inline 關鍵字

Kotlin 代碼示例
inline fun inlineFunction(block: () -> Unit) {println("Before block")block()println("After block")
}fun main() {inlineFunction {println("Inside block")}
}
編譯后代碼分析

在編譯時,inline 函數會被內聯展開。上述代碼編譯后,大致等效于以下 Java 代碼(Kotlin 編譯成 JVM 字節碼,這里用 Java 形式便于理解):

public class Main {public static void main(String[] args) {System.out.println("Before block");System.out.println("Inside block");System.out.println("After block");}
}

inlineFunction 的函數體直接替換了調用處的代碼,避免了函數調用的開銷。

2. noinline 關鍵字

Kotlin 代碼示例
inline fun mixedFunction(inlineBlock: () -> Unit, noinline noInlineBlock: () -> Unit) {inlineBlock()noInlineBlock()
}fun main() {mixedFunction({ println("Inline block") },{ println("Noinline block") })
}
編譯后代碼分析

inlineBlock 會被內聯展開,而 noInlineBlock 不會。編譯后的代碼大致如下:

public class Main {private static final class NoInlineBlock implements Function0<Unit> {public Unit invoke() {System.out.println("Noinline block");return Unit.INSTANCE;}}public static void main(String[] args) {System.out.println("Inline block");NoInlineBlock noInlineBlock = new NoInlineBlock();noInlineBlock.invoke();}
}

inlineBlock 被內聯到調用處,而 noInlineBlock 被封裝成一個實現了 Function0 接口的類,通過創建對象并調用 invoke 方法來執行。

3. crossinline 關鍵字

Kotlin 代碼示例
inline fun crossInlineFunction(crossinline block: () -> Unit) {val wrapper = {block()}wrapper()
}fun main() {crossInlineFunction {println("Crossinline block")}
}
編譯后代碼分析

crossinline 保證了 Lambda 表達式不會有非局部返回,但仍然會被內聯。編譯后的代碼大致如下:

public class Main {public static void main(String[] args) {final class Wrapper implements Function0<Unit> {public Unit invoke() {System.out.println("Crossinline block");return Unit.INSTANCE;}}Wrapper wrapper = new Wrapper();wrapper.invoke();}
}

block 被內聯到 wrapper 中,同時由于 crossinline 的存在,避免了非局部返回的問題。

總結來說,inline 關鍵字使函數體在調用處展開,noinline 阻止特定函數參數內聯,crossinline 允許內聯的同時限制非局部返回,這些關鍵字在編譯后的代碼中體現了不同的處理方式。

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/diannao/74276.shtml
繁體地址,請注明出處:http://hk.pswp.cn/diannao/74276.shtml
英文地址,請注明出處:http://en.pswp.cn/diannao/74276.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

LabVIEW FPGA與Windows平臺數據濾波處理對比

LabVIEW在FPGA和Windows平臺均可實現數據濾波處理&#xff0c;但兩者的底層架構、資源限制、實時性及應用場景差異顯著。FPGA側重硬件級并行處理&#xff0c;適用于高實時性場景&#xff1b;Windows依賴軟件算法&#xff0c;適合復雜數據處理與可視化。本文結合具體案例&#x…

深度解析 Android Matrix 變換(二):組合變換 pre、post

前言 在上一篇文章中&#xff0c;我們講解了 Canvas 中單個變換的原理和效果&#xff0c;即縮放、旋轉和平移。但是單個旋轉僅僅是基礎&#xff0c;Canvas 變換最重要的是能夠隨意組合各種變換以實現想要的效果。在這種情況下&#xff0c;就需要了解如何組合變換&#xff0c;以…

Java并發編程之CountDownLatch

1. 基本原理 計數器 CountDownLatch 在創建時需要指定一個初始計數值。這個值通常代表需要等待完成的任務數或線程數。 等待與遞減 等待&#xff1a;調用 await() 方法的線程會被阻塞&#xff0c;直到計數器變為 0。遞減&#xff1a;每當一個任務完成后&#xff0c;應調用 cou…

C++|GLog開源庫的使用 如何實現自定義類型消息日志

參考&#xff1a; C glog使用教程與代碼演示 C第三方日志庫Glog的安裝與使用超詳解 GLOG從入門到入門 glog 設置日志級別_glog C版本代碼分析 文章目錄 日志等級自定義消息創建使用宏定義 日志等級 在 glog 中&#xff0c;日志的嚴重性是通過 LogSeverity 來區分的&#xff0c…

FAQ - VMware vSphere Web 控制臺中鼠標控制不了怎么辦?

問題描述 在VMware vSphere vCenter Server 的 Web 控制臺中新建了一臺 Windows Server 2008 R2 虛擬機&#xff0c;但是鼠標進入控制臺后&#xff0c;可以看見鼠標光標&#xff0c;但是移動卻沒有反應。 根因分析 暫無。 解決方案 選中虛擬機>操作>編輯設置>添加新…

Rust+WebAssembly:開啟瀏覽器3D渲染新時代

引言 在當今的 Web 開發領域&#xff0c;隨著用戶對網頁交互體驗的要求日益提高&#xff0c;3D 渲染技術在 Web 應用中的應用愈發廣泛。從沉浸式的 Web 游戲&#xff0c;到逼真的虛擬展示場景&#xff0c;3D 渲染引擎承擔著將虛擬 3D 世界呈現到用戶瀏覽器中的關鍵任務。其性能…

在小米AX6000中添加tailscale monitor

經過測試&#xff0c;發現小米路由器中的tailscale可能會因為某種原因狀態異常&#xff0c; 為了讓tailscale恢復正常&#xff0c;所以又寫了monitor用來監控&#xff1a; #!/bin/sh# Define Tailscale related paths TAILSCALED_PATH"/tmp/tailscale/tailscale_1.80.3_a…

表達式括號匹配(stack)(信息學奧賽一本通-1353)

【題目描述】 假設一個表達式有英文字母&#xff08;小寫&#xff09;、運算符&#xff08;&#xff0c;—&#xff0c;?&#xff0c;/&#xff09;和左右小&#xff08;圓&#xff09;括號構成&#xff0c;以“ ”作為表達式的結束符。請編寫一個程序檢查表達式中的左右圓括號…

IM 基于 WebRtc 視頻通信功能

IM&#xff08;即時通訊&#xff09;基于 WebRTC&#xff08;Web Real-Time Communication&#xff0c;網頁實時通訊&#xff09; 原理 WebRTC 是一種支持網頁瀏覽器進行實時語音通話或視頻通話的技術&#xff0c;它提供了一組 JavaScript API&#xff0c;使得在瀏覽器之間無…

關于極端場景下,數據庫更新與 MQ 消息一致性保障方案的詳細總結

目錄 一、核心問題場景 二、RocketMQ 事務消息方案 1. 核心機制 2. 執行流程 3. 關鍵優勢 4. 局限性 三、消息表方案 1. 核心機制 2. 執行流程 3. 關鍵優勢 4. 局限性 四、方案對比與選擇 五、實施建議 六、總結 一、核心問題場景 當數據庫更新后,若 MQ 消息未…

【設計模式】單件模式

七、單件模式 單件(Singleton) 模式也稱單例模式/單態模式&#xff0c;是一種創建型模式&#xff0c;用于創建只能產生 一個對象實例 的類。該模式比較特殊&#xff0c;其實現代碼中沒有用到設計模式中經常提起的抽象概念&#xff0c;而是使用了一種比較特殊的語法結構&#x…

【redis】主從復制:拓撲結構、原理和psync命令解析

文章目錄 拓撲一主一從相關問題 一主多從相關問題 樹形主從結構相關問題 主從復制原理復制流程 psync 命令命令解析replicatonidoffset總結 運行流程 拓撲 若干個節點之間按照什么樣的方式來進行組織連接 一主一從 都可以讀&#xff0c;從節點可以幫主節點分擔一部分的壓力只…

[RoarCTF 2019]Easy Calc-3.23BUUCTF練習day5(2)

[RoarCTF 2019]Easy Calc-3.23BUUCTF練習day5(2) 解題過程 查看源碼 發現calc.php頁面&#xff0c;訪問一下 分析代碼 首先獲取$_GET[num]的值并賦給變量$str。然后定義了一個黑名單數組$blacklist&#xff0c;包含了一系列被禁止的字符或轉義字符&#xff0c;如空格、制表…

阻塞隊列:原理、應用及實現

阻塞隊列&#xff1a;原理、應用及實現 什么是阻塞隊列以生產消費者模型形象地理解阻塞隊列阻塞隊列實現生產消費者模型模擬實現阻塞隊列實現生產消費者模型 什么是阻塞隊列 阻塞隊列是一種特殊且實用的隊列數據結構&#xff0c;它同樣遵循 “先進先出” 的原則。與普通隊列不…

【開源寶藏】30天學會CSS - DAY5 第五課 脈沖動畫

以下是一個完整的漸進式教程&#xff0c;拆解如何用 HTML CSS 構建“Pulsar”水波脈沖動畫。通過閱讀&#xff0c;你將理解每個核心屬性與關鍵幀如何配合&#xff0c;讓一個小圓不斷散發動態波紋&#xff0c;并且文字始終停留在圓心。 第 0 步&#xff1a;項目概覽 文件結構示…

2060 裁紙刀

2060 裁紙刀 ??難度&#xff1a;簡單 &#x1f31f;考點&#xff1a;2022、規律、思維 &#x1f4d6; &#x1f4da; import java.util.Arrays; import java.util.LinkedList; import java.util.Queue; import java.util.Scanner;public class Main {static int N 100010…

TextView、AppCompatTextView和MaterialTextView該用哪一個?Android UI 組件發展史與演進對照表

在 Android 開發中&#xff0c;UI 組件一直在不斷演進&#xff0c;從最初的原生組件&#xff0c;到 Support Library&#xff08;AppCompat 兼容庫&#xff09;&#xff0c;再到如今的 Material Design 組件。這篇文章將梳理 Android UI 組件的發展歷史&#xff0c;并提供詳細的…

python學習筆記--實現簡單的爬蟲(一)

任務&#xff1a;爬取豆瓣最受歡迎的250個電影的資料 鏈接&#xff1a;豆瓣電影 Top 250 用瀏覽器打開后&#xff0c;使用F12或鼠標右鍵--檢查&#xff0c;查看網頁的源代碼&#xff0c;分析網頁結構&#xff0c;如下圖所示&#xff1a; 分析后得知&#xff1a; 1.電影名位于…

Postgresql 刪除數據庫報錯

1、刪除數據庫時&#xff0c;報錯存在其他會話連接 ## 錯誤現象&#xff0c;存在其他的會話連接正在使用數據庫 ERROR: database "cs" is being accessed by other users DETAIL: There is 1 other session using the database.2、解決方法 ## 終止被刪除數據庫下…

self Attention為何除以根號dk?(全新角度)

全網最獨特解析&#xff1a;self Attention為何除根號dk&#xff1f; 一、假設條件&#xff1a;查詢向量和鍵向量服從正態分布 假設查詢向量 q i q_i qi?和鍵向量 k j k_j kj?的每個分量均為獨立同分布的隨機變量&#xff0c;且服從標準正態分布&#xff0c;即&#xff1a;…