[kotlin] inline 函数

inline

inline函数是什么?

  • nline 函数,就是在编译时会把函数体和传入的 lambda 代码"展开"到调用处,而不是像普通函数一样生成函数调用。
    这样做的最大特点是:消除了高阶函数调用的性能损耗

inline函数解决了什么问题?

问题:高阶函数的性能损耗

  • 每次传 lambda,JVM 都会创建一个对象(匿名类),并有额外的函数调用开销。如果高阶函数用得多,性能损耗就明显了。
kotlin 复制代码
fun doSomething(action: () -> Unit) { action() }

解决方案:inline函数

  • Kotlin 的 inline 关键字告诉编译器:
    把函数体和传入的 lambda 代码直接展开到调用处,不用生成额外对象和函数调用。 这样 lambda 不再生成对象,也没有函数调用栈,性能几乎和普通代码一样高效。
kotlin 复制代码
inline fun doSomething(action: () -> Unit) { action() }

典型应用场景

  • 集合操作 (如 forEachmapfilter 等)都用 inline,提高性能。
  • DSL 设计:比如 Kotlin 的协程、Anko、Jetpack Compose 等,广泛用 inline 实现语法糖。

缺点

  • 代码膨胀(Code Bloat) :每次调用 inline 函数,都会把代码"展开",如果调用很多次,最终编译出的字节码会变大,影响 APK/程序体积。
  • 调试困难:因为函数被展开,调试时栈信息变得不直观。
  • 不能用递归:inline 函数不能递归调用自己,否则展开会无限循环。
  • 某些场景不能 inline :比如 lambda 被跨作用域保存(这时需要用 noinlinecrossinline 修饰)。

crossinline

crossinline

Kotlin 中用于 lambda 参数声明 的一个修饰符,主要用于 inline 函数

它的作用是:禁止在传入的 lambda 表达式中使用 non-local return ,即不能在 lambda 里直接用 return 跳出外层函数。

  • 场景 代码
kotlin 复制代码
inline fun foo(action: () -> Unit) {
    action() // 如果 action() 里 return,会直接退出 foo
}
fun test() {
    foo {
        return // 非本地 return,直接退出 test()
    }
}

crossinline 的作用

有些情况下,这种"非本地 return"是不安全或不允许的,比如你要把 lambda 存起来、用在另一个线程、或者作为参数传递。

这时就需要用 crossinline 修饰 lambda 参数,强制只允许局部 return(即只能退出 lambda 本身,不能跳出外层函数)。

kotlin 复制代码
inline fun doSomething(crossinline action: () -> Unit) {
    val runnable = Runnable { action() }
    runnable.run()
}
fun test() {
    doSomething {
        // return // 编译错误!不能在 crossinline lambda 里用非本地 return
        println("hello")
    }
}
相关推荐
神奇的程序员14 小时前
我的软件冲进苹果商店下载榜前 50 了
前端
阳光是sunny14 小时前
别再被 worktree 绕晕了!AI 编程时代你必须掌握的 Git 隔离神器
前端·人工智能·后端
万少16 小时前
万少的博客 - 技术分享与解决方案
前端·javascript·后端
尘世中一位迷途小书童18 小时前
用 Cesium 撸了一个森林火情监控大屏,弧线、粒子、发光效果都齐了
前端·javascript
IT_陈寒19 小时前
垃圾回收器选错了,我的Java服务内存炸了
前端·人工智能·后端
月光下的丝瓜19 小时前
Flutter 国内安装指南
前端·flutter
玄星啊20 小时前
AI 编程的第 30 天,我怀念古法 Coding 了
前端·ai编程
Jolyne_20 小时前
Angular基础速通
前端·angular.js
锋行天下21 小时前
半秒开!还有谁!!!
前端·vue.js·架构
代码搬运媛1 天前
git 下中文文件名乱码问题解决
前端