当前位置: 首页 > backend >正文

Kotlin 内联函数

Kotlin 的 内联函数(inline function) 是一种优化手段,它允许在编译时将函数的代码“展开”到调用处,从而避免函数调用的开销,尤其适用于 高阶函数(接受函数作为参数) 的场景。

什么是内联函数

正常函数调用会产生函数对象、栈帧和跳转指令,但内联函数会在编译时将函数代码插入到调用点,避免这些开销。

inline fun doSomething(block: () -> Unit) {println("Before")block()println("After")
}fun main() {doSomething {println("Hello")}
}

编译后大致等价于:

fun main() {println("Before")println("Hello")println("After")
}

函数体会“复制粘贴”到调用点。

什么时候使用 inline

Kotlin 中高阶函数(比如 letrunwithapply)频繁使用内联,避免函数对象的创建(block: () -> Unit 不会生成 Function 对象),避免 lambda 的闭包捕获、堆分配和 GC 开销。普通业务函数不推荐使用内联,会导致代码膨胀。

inline 的工作原理

内联函数不是运行时特性,而是编译器替换机制inline fun foo(block: () -> Unit) 编译器看到后就会展开 block() 的实现到调用点。如果有多个调用点,会展开多次(可能造成代码膨胀)。

使用 return@label可以局部返回:

inline fun doSomething(block: () -> Unit) {block()
}fun main() {doSomething {println("Inside block")return@doSomething // 局部返回,不影响 main}println("After")
}

如果不加 inline 会怎么样,编译器会生成 Function 实例(Function0),lambda 体被封装为对象,可能造成性能下降,尤其是在 Android 中。

fun doSomething(block: () -> Unit) {block()
}

noinline 与 crossinline

noinline

默认情况下,所有函数参数都会被内联。但你可以用 noinline 排除部分参数:

inline fun test(block1: () -> Unit, noinline block2: () -> Unit) {block1()  // 被内联block2()  // 作为函数对象
}

noinline 主要用于函数参数需要作为对象传递,以及不能内联的上下文(如存储到变量)。

inline fun doOps(inlineBlock: () -> Unit,noinline logging: () -> Unit  // 这里阻止了 logging 被内联
) {inlineBlock()   // 内联val func = logging  // 如果不加 noinline 会编译错误,因为不能传递被内联的 lambdafunc()
}

crossinline

禁止 lambda 用 return 做“非局部返回”。默认情况下,内联 lambda 可以直接 return,但有风险。

inline fun higherOrder(block: () -> Unit) {println("Before")block()println("After") // 如果 block 里直接 return,这行永远不会执行
}fun test() {higherOrder {return  // 这是“非局部返回”:会从 test() 函数直接返回}
}

要禁止这种“跳出上层函数”的行为,加 crossinline

inline fun higherOrder(crossinline block: () -> Unit) {println("Before")block()println("After") // 如果 block 里直接 return,这行永远不会执行
}fun test() {higherOrder {return  // 现在 block() 里面不能写 return 了,这里会报错,但是依然可以写 return@higherOrder}
}

注意事项与风险

代码膨胀

inline fun heavy(block: () -> Unit) { ... }repeat(1000) {heavy { ... } // 编译后生成 1000 份代码
}

函数体复杂,调用频繁时,会导致APK 变大,方法数上升。

调试困难

内联函数展开后,调试栈信息会变得混乱,调试时跳转不准。

http://www.xdnf.cn/news/16198.html

相关文章:

  • LeetCode|Day25|389. 找不同|Python刷题笔记
  • 小程序安卓ApK转aab文件详情教程MacM4环境
  • C++中std::string和std::string_view使用详解和示例
  • Redis数据库入门教程
  • 前端安全问题怎么解决
  • 一篇文章了解HashMap和ConcurrentHashMap的扩容机制
  • Node.js 中的内置模板path
  • 论文阅读:《Many-Objective Evolutionary Algorithms: A Survey. 》多目标优化问题的优化目标评估的相关内容介绍
  • 机器翻译编程
  • 【安卓笔记】解决livedata粘性事件
  • 在 Alpine Linux 中创建虚拟机时 Cgroup 挂在失败的现象
  • Springboot宠物用品商城的设计与实现
  • 详解力扣高频SQL50题之197. 上升的温度【简单】
  • 星慈光编程虫2号小车讲解第二篇--向左向右平移
  • Python编程进阶知识之第五课处理数据(matplotlib)
  • Unity VS Unreal Engine ,“电影像游戏的时代” 新手如何抉择引擎?(结)
  • 100条SQL语句分类精讲:从基础到进阶的实操指南
  • 医疗系统国产化实录:SQL Server国产替代,乙方保命指南
  • 机器学习的基础知识
  • 洛谷 P1996 约瑟夫问题之题解
  • kafka的shell操作
  • 多源信息融合智能投资【“图神经网络+强化学习“的融合架构】【低配显卡正常运行】
  • MapStruct类型转换接口未自动注入到spring容器中
  • 快速将前端得依赖打为tar包(yarn.lock版本)并且推送至nexus私有依赖仓库(笔记)
  • 《C++》面向对象编程--类(下)
  • LLM中的位置嵌入矩阵(Position Embedding Matrix)是什么
  • matrix-breakout-2-morpheus靶机通关教程
  • DBA常用数据库查询语句
  • Python爬虫案例:Scrapy+XPath解析当当网网页结构
  • Lua(模块与包)