Kotlin 中的 inline 和 reified 关键字

在 Kotlin 里,inline 和 reified 经常成对出现,但它们各自解决的问题并不一样。下面把两个关键字拆开讲,再给出它们"合体"时的典型用法。


  1. inline ------ 消除"函数对象"的开销

背景

Kotlin 的高阶函数(参数里带 lambda)会被编译成对象。每次调用都生成一个匿名类/对象,带来:

  • 内存分配
  • 虚方法调用
  • 运行时实例化

inline 做了什么

函数体 以及传给它的 lambda 字节码原地展开(copy-paste),彻底消除上述开销。

示例

kotlin 复制代码
// 未 inline
fun measure(block: () -> Unit) {
    val t0 = System.nanoTime()
    block()
    println("cost ${System.nanoTime() - t0} ns")
}

// 调用
measure { println("hello") }

反编译后能看到 measure 里出现了 Function0 实例。

加上 inline:

kotlin 复制代码
inline fun measure(block: () -> Unit) {
    val t0 = System.nanoTime()
    block()
    println("cost ${System.nanoTime() - t0} ns")
}

反编译结果里只剩下一行 println("hello") 和计时代码,没有任何 Function0

额外效果

  • return 可以"非局部"返回(lambda 里直接返回外层函数)。
  • 整个调用栈在调试器里变"平",断点会停在展开后的行号。

  1. reified ------ 把"擦除"的类型"找回来"

背景

JVM 泛型在运行时会被擦除,因此下面代码编译失败:

kotlin 复制代码
fun <T> Bundle.get(): T? = this.getSerializable("key") as? T   // 警告:unchecked cast

问题

运行时无法判断 T 具体是什么,因而无法安全转换。

reified 做了什么

只有 inline 函数才能加 reified;它把类型实参 也写进字节码里,于是运行时就能拿到 Class 对象。

示例

kotlin 复制代码
inline fun <reified T> Bundle.get(): T? =
    if (T::class.java.isPrimitive) null
    else this.getSerializable("key") as? T

调用

kotlin 复制代码
val user: User? = bundle.get<User>()

反编译后能看到:

java 复制代码
User $tmp = (User) bundle.getSerializable("key");

User.class 被直接硬编码在调用处,因此运行期可以安全强转。


  1. 合体:inline + reified 的典型套路

  1. 启动 Activity(Android)
kotlin 复制代码
inline fun <reified T : Activity> Context.startActivity() {
    startActivity(Intent(this, T::class.java))
}

调用:

kotlin 复制代码
startActivity<DetailActivity>()
  1. 获取日志 Tag
kotlin 复制代码
inline fun <reified T> T.logger(): Logger = LoggerFactory.getLogger(T::class.java)
  1. Gson 一行反序列化
kotlin 复制代码
inline fun <reified T> Gson.fromJson(json: String): T = fromJson(json, T::class.java)

  1. 小结一句话

  • inline性能+控制流关键字:消除 lambda 对象、允许非局部返回。
  • reified类型信息关键字:只能配合 inline,让泛型类型在运行期"不擦除"。
相关推荐
无风之翼13 小时前
android15 休眠唤醒过程中有时候屏幕显示时间一闪而过
android·锁屏
yue00813 小时前
C# winform自定义控件
开发语言·c#
JANGHIGH14 小时前
c++ 多线程(三)
开发语言·c++
侠***I14 小时前
基于OOA-TCN-BiGRU-Attention的鱼鹰算法优化多变量时间序列预测
kotlin
2503_9284115614 小时前
12.9 Vue3+Vuex+Js+El-Plus+vite(项目搭建)
开发语言·javascript·ecmascript
卓码软件测评14 小时前
第三方软件验收评测机构【Gatling安装指南:Java环境配置和IDE插件安装】
java·开发语言·ide·测试工具·负载均衡
weixin_3077791314 小时前
Jenkins中的Jakarta Activation API插件:功能、使用与最佳实践
运维·开发语言·ci/cd·自动化·jenkins
weixin_4399306414 小时前
前端js日期计算跨月导致的错误
开发语言·前端·javascript
方白羽15 小时前
Android全局悬浮拖拽视图
android·app·客户端
liu****15 小时前
9.二叉树(一)
c语言·开发语言·数据结构·算法·链表