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

Kotlin 中的 inline 和 reified 关键字

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


  1. inline —— 消除“函数对象”的开销

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

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

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

示例

// 未 inline
fun measure(block: () -> Unit) {val t0 = System.nanoTime()block()println("cost ${System.nanoTime() - t0} ns")
}// 调用
measure { println("hello") }

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

加上 inline:

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

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

额外效果

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

  1. reified —— 把“擦除”的类型“找回来”

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

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

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

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

示例

inline fun <reified T> Bundle.get(): T? =if (T::class.java.isPrimitive) nullelse this.getSerializable("key") as? T

调用

val user: User? = bundle.get<User>()

反编译后能看到:

User $tmp = (User) bundle.getSerializable("key");

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


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

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

调用:

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

  1. 小结一句话

  • inline性能+控制流关键字:消除 lambda 对象、允许非局部返回。
  • reified类型信息关键字:只能配合 inline,让泛型类型在运行期“不擦除”。
http://www.dtcms.com/a/605037.html

相关文章:

  • 开封府景点网站及移动端建设情况精品资源共享课网站建设 碧辉腾乐
  • 战场目标检测:Faster R-CNN与RegNetX-800MF融合实现建筑物人员坦克车辆识别_2
  • 易语言黑月编译器:提升编程效率与性能优化 | 深入解析易语言开发中的工具应用与技巧
  • Vibe Coding - 从Vibe Coding到Spec Coding_AI编码范式的进化之路
  • 宣化网站建设青岛网站制作推广平台
  • 【多模态大模型面经】 BERT 专题面经
  • Node.js 开发实战:从入门到精通
  • 草莓病害智能识别与分类_Cascade-RCNN_HRNetV2p-W18-20e_COCO实现
  • 改造多模块!!无法使用三方依赖的异常处理
  • JMeter 自动化实战:自动生成文件并传参接口的完整方案
  • AutoSAR实战:RTA-OS Counters操作系统计数器详解
  • FCAF3D: Fully Convolutional Anchor-Free 3D Object Detection论文精读
  • 北京市轨道交通建设管理有限公司网站企业网站建设合同书模板
  • 做图表的网站大连关键词
  • Vue 3中集成GIS(地理信息系统)
  • 进程基本概念
  • Java模拟算法题目练习
  • Mac远程控制新篇章:UU远程被控端深度测评
  • WordPress插件--菜单登录后可见的插件
  • 电商数据分析报告
  • Rust与主流编程语言客观对比:特性、场景与实践差异
  • C语言编译器有哪些 | 选择最适合的编译器提高开发效率
  • 网站频道规划网站个人备案模版
  • 昆明公司建设网站制作上海seo外包
  • MySQL: 存储引擎选择策略:基于事务支持、备份需求、崩溃恢复及特性兼容性的综合指南
  • 学生成绩管理系统 基于java+springboot+vue实现前后端分离项目并附带万字文档(源码+数据库+万字详设文档+软件包+安装教程)
  • ios-WebP
  • 网站如何做网站解析品牌策划方案怎么做
  • 能源企业合同管理数智化转型解决方案
  • 国标新规下的零碳园区新基建:碳电表与智慧能源平台的协同赋能