Kotlin Gradle plugin 中的编译与缓存
在本章中, 你将学习以下内容:
增量编译(Incremental compilation)
Kotlin Gradle plugin 支持增量编译模式. 增量编译模式会监视 classpath 中的文件在两次编译之间的变更, 因此只有变更过的文件会被编译. 增量编译与 Gradle 的构建缓存 一起配合工作, 并支持 编译回避(compilation avoidance).
增量编译模式支持 Kotlin/JVM 和 Kotlin/JS 工程, 并且默认开启.
有以下几种方式可以禁用增量编译设定:
对 Kotlin/JVM 项目: 设置
kotlin.incremental=false
.对 Kotlin/JS 项目: 设置
kotlin.incremental.js=false
.在命令行参数中, 添加
-Pkotlin.incremental=false
或-Pkotlin.incremental.js=false
.需要向所有后续的编译命令都添加这个参数.
注意: 任何一次编译如果关闭了增量编译模式, 都会导致增量编译的缓存失效. 初次编译不会是增量编译.
如果你想要了解我们目前的增量编译方案如何工作, 以及与以前方案的区别, 请阅读我们的 blog.
对编译任务的输出的精确备份
从 Kotlin 1.8.20 开始, 你可以启用精确备份功能, 这时只有 Kotlin 在增量编译中重新编译的那些类会被备份. 完整备份和精确备份都可以帮助在发生编译错误后再次运行增量构建. 精确备份与完整备份相比, 会耗费较少的构建时间. 对于大型的项目, 或者很多任务都创建备份, 那么完整备份可能会花费 明显 更长的构建时间, 尤其是如果项目位于速度较慢的 HDD 上.
要启用这个优化功能, 请向 gradle.properties
文件添加 kotlin.compiler.preciseCompilationResultsBackup
Gradle 属性:
JetBrains 使用精确备份的例子
在下面的图表中, 你可以看到使用精确备份与完整备份相对比的示例:
第一个和第二个对比图显示了在 Kotlin 项目中使用精确备份时对 Kotlin Gradle plugin 构建的影响:
进行一个小的 ABI 变更之后: 向一个被大量模块依赖的模块添加一个新的 public 方法.
进行一个小的非 ABI 变更之后: 向一个没有被其他模块依赖的模块添加一个 private 函数.
第三个对比图显示了在 Space 项目中使用精确备份时, 在小的非 ABI 更改后对 Web 前端构建的影响: 向一个被大量模块依赖的 Kotlin/JS 模块添加一个 private 函数.
我们在使用 Apple M1 Max CPU 的计算机上进行这些测量; 在不同的计算机上会出现稍微不同的结果. 影响性能的因素包括但不限于以下几点:
Kotlin daemon 和 Gradle daemon 热身状况(warm)如何.
硬盘速度如何.
CPU 型号, 以及它的繁忙程度.
哪些模块受到变更的影响, 以及这些模块有多大.
是 ABI 变更还是非 ABI 变更.
使用构建报告来评估优化
要对你的项目和场景, 评估优化在你的计算机上的影响, 你可以使用 Kotlin 构建报告. 请向你的 gradle.properties
文件添加下面的属性, 启用文本文件格式的构建报告:
下面是在启用精确备份之前, 构建报告的相关部分的示例:
下面是在启用精确备份之后, 构建报告的相关部分的示例:
对 Gradle 构建缓存的支持
Kotlin 插件支持 Gradle 构建缓存, 构建缓存会保存构建的输出, 并在未来的构建中重复使用.
如果想要对所有的 Kotlin 任务禁用缓存, 请将系统属性 kotlin.caching.enabled
设置为 false
(也就是使用参数 -Dkotlin.caching.enabled=false
来执行编译).
对 Gradle 配置缓存的支持
Kotlin plugin 使用 Gradle 配置缓存, 通过对之后的构建重用配置阶段的结果, 来增加构建处理的速度.
关于如何启用配置缓存, 请参见 Gradle 文档. 启用这个功能之后, Kotlin Gradle plugin 会自动开始使用它.
Kotlin daemon 及其在 Gradle 中的使用
Kotlin daemon 会:
与 Gradle daemon 共同运行来编译项目.
当你使用 IntelliJ IDEA 内建的构建系统来编译项目时, 它会在 Gradle daemon 之外单独运行.
在 Gradle 的 执行阶段, 当一个 Kotlin 编译任务开始编译源代码时, Kotlin daemon 就会启动. Kotlin daemon 会和 Gradle daemon 一起停止, 或在没有 Kotlin 编译任务执行, 空闲 2 个小时之后停止.
Kotlin daemon 使用与 Gradle daemon 相同的 JDK.
设置 Kotlin daemon 的 JVM 参数
以下列表是设置参数的几种不同方式, 列表中设置方式按照优先级排列, 每种方式都会覆盖在它之前的其他方式:
继承 Gradle daemon 参数
如果不做任何设定, Kotlin daemon 会从 Gradle daemon 继承 JVM 参数. 比如, 在 gradle.properties
文件中:
设置系统属性 kotlin.daemon.jvm.options
如果 Gradle daemon 的 JVM 参数包含 kotlin.daemon.jvm.options
系统属性 – 请在 gradle.properties
文件中指定:
传递参数时, 要遵守以下规则:
只有 在参数
Xmx
,XX:MaxMetaspaceSize
, 和XX:ReservedCodeCacheSize
之前要使用减号-
, 在其它参数之前不要使用.参数之间的分隔使用逗号 (
,
), 不带空格. 空格之后的参数会被 Gradle daemon 使用, 而不是被 Kotlin daemon 使用.
设置属性 kotlin.daemon.jvmargs
你可以在 gradle.properties
文件中添加 kotlin.daemon.jvmargs
属性:
使用 kotlin 扩展
你可以在 kotlin
扩展中指定参数:
使用特定的任务定义
你可以对特定的任务指定参数:
指定 JVM 参数时 Kotlin daemon 的行为
配置 Kotlin daemon 的 JVM 参数时, 请注意:
如果不同的子项目或任务设置了不同的 JVM 参数, 那么会存在多个 Kotlin daemon 实例同时运行.
只有当 Gradle 运行相关的编译任务, 而且现存的 Kotlin daemon 实例没有使用相同的 JVM 参数时, 才会启动新的 Kotlin daemon 实例. 假设你的项目包含很多子项目. 大部分子项目对 Kotlin daemon 需要某种 heap memory 设定, 但有一个模块需要很大的 heap memory 设定 (然而这个模块很少被编译). 这种情况下, 你应该对这个模块设定不同的 JVM 参数, 这样, 就可以只有在开发者编译这个特定模块时, 才会使用很大的 heap memory 启动一个 Kotlin daemon.
如果
Xmx
参数未指定, Kotlin daemon 会从 Gradle daemon 继承.
回退到以前的编译器
从 Kotlin 2.0.0 开始, 默认使用 K2 编译器.
要在 Kotlin 2.0.0 之后的版本中使用以前的编译器, 请使用以下方法:
在你的
build.gradle.kts
文件中, 设置语言版本 为1.9
.或者
使用以下编译器选项:
-language-version 1.9
.
关于 K2 编译器的优点, 请参见 K2 编译器迁移向导.
定义 Kotlin 编译器执行策略
Kotlin 编译器执行策略 定义 Kotlin 编译器在哪里执行, 以及各种情况下是否支持增量编译.
有 3 种编译器执行策略:
策略 | Kotlin 编译器在哪里执行 | 增量编译 | 其它特征, 以及注意事项 |
---|---|---|---|
Daemon | 在 Kotlin 自己的 daemon 进程之内 | 是 | 默认的, 而且最快的策略. 可以在不同的 Gradle daemon, 以及多个并行编译之间共用. |
In process | 在 Gradle daemon 进程之内 | 否 | 可以与 Gradle daemon 共用 heap. "In process" 执行策略比 "Daemon" 执行策略 更慢. 每个 worker 会为每个编译创建单独的 Kotlin 编译器 classloader. |
Out of process | 对每个编译都在单独的进程内 | 否 | 这是最慢的执行策略. 与 "In process" 类似, 但还会为每个编译在 Gradle worker 内创建单独的 Java 进程. |
要定义一个 Kotlin 编译器执行策略, 你可以使用以下属性之一:
Gradle 属性
kotlin.compiler.execution.strategy
.Compile Task 属性
compilerExecutionStrategy
.
Task 属性 compilerExecutionStrategy
的优先级高于 Gradle 属性 kotlin.compiler.execution.strategy
.
kotlin.compiler.execution.strategy
属性可以使用的值是:
daemon
(默认值)in-process
out-of-process
在 gradle.properties
中使用 Gradle 属性 kotlin.compiler.execution.strategy
:
Task 属性 compilerExecutionStrategy
可以使用的值是:
org.jetbrains.kotlin.gradle.tasks.KotlinCompilerExecutionStrategy.DAEMON
(默认值)org.jetbrains.kotlin.gradle.tasks.KotlinCompilerExecutionStrategy.IN_PROCESS
org.jetbrains.kotlin.gradle.tasks.KotlinCompilerExecutionStrategy.OUT_OF_PROCESS
在你的构建脚本中使用 Task 属性 compilerExecutionStrategy
:
Kotlin 编译器的 fallback 策略
Kotlin 编译器的 fallback 策略是指, 如果 daemon 因为某种原因失败, 则在 Kotlin daemon 之外运行编译任务. 如果 Gradle daemon 启动, 编译器会使用 "In process" 策略. 如果 Gradle daemon 没有启动, 编译器会使用 "Out of process" 策略.
当 fallback 发生时, 在你的 Gradle 构建输出中会收到以下警告信息:
但是, 静默的 fallback 到其他策略, 会消耗大量的系统资源, 或导致不确定的构建结果. 关于这个问题, 请参见这个 YouTrack issue. 要避免这个问题, 有一个 Gradle 属性 kotlin.daemon.useFallbackStrategy
, 默认值为 true
. 当它的值设置为 false
时, daemon 启动或通信时问题会导致构建失败. 请在 gradle.properties
文件中声明这个属性:
在 Kotlin 编译任务中也有一个 useDaemonFallbackStrategy
属性, 如果同时使用, 它的优先级会高于 Gradle 属性.
如果运行编译所需要的内存不足, 你会在 log 中看到相关信息.
试用最新的语言版本
从 Kotlin 2.0.0 开始, 要试用最新的语言版本, 请在你的 gradle.properties
文件中设置 kotlin.experimental.tryNext
属性. 使用这个属性时, Kotlin Gradle plugin 会将语言版本增加到比你的 Kotlin 版本的默认值的下一个版本. 例如, 在 Kotlin 2.0.0 中, 默认的语言版本是 2.0, 因此这个属性会将语言版本配置为 2.1.
或者, 你也可以运行以下命令:
在 构建报告 中, 你可以看到用来编译每个任务的语言版本.
构建报告
构建报告包括不同编译阶段的持续时间, 以及为什么不能进行增量编译的原因. 如果编译时间太长, 或对于相同的项目出现了不同的编译时间, 可以使用构建报告来调查性能问题.
Kotlin 构建报告可以帮助你调查构建性能相关的问题, 它比 Gradle build scans 更加有效, Gradle Build Scan 中的粒度只是单个 Gradle Task.
通过对长时间运行的编译分析构建报告, 可以帮助你解决两种常见问题:
构建不能增量模式运行. 分析原因, 并解决底层问题.
构建是增量模式运行, 但耗费太多时间. 可以尝试重新组织源代码文件 — 切分大的文件, 将不同的类保存到不同的文件, 重构大的类, 在不同的文件中声明顶层函数, 等等.
构建报告还会显示项目中使用的 Kotlin 版本. 此外, 从 Kotlin 1.9.0 开始, 你可以在你的 Gradle Build Scan 中看到, 编译代码时使用的是哪个编译器.
请参见 如何阅读构建报告 以及 JetBrains 如何使用构建报告.
启用构建报告
要启用构建报告, 请在 gradle.properties
中指定构建报告输出的保存位置:
以下各个值的组合可以用于输出:
选项 | 含义 |
---|---|
| 将构建报告保存到本地文件, 使用可供人类阅读的格式. 默认设置是 |
| 将构建报告保存到本地文件, 使用二进制对象格式. |
| 将构建报告保存到 build scan 的 |
| 通过 HTTP(S) 提交构建报告. 使用 POST 方法传送 JSON 格式的测量结果. 你可以在 Kotlin 代码仓库 中看到传送的数据的当前版本. 你可以在 这篇 Blog 看到 HTTP Endpoint 的示例 |
| 将构建报告保存到本地文件, 使用 JSON 格式. 请使用 |
下面是 kotlin.build.report
的选项列表:
只适用于 HTTP 输出的选项:
custom values 的限制
为了收集 build scan 的统计信息, Kotlin 构建报告会使用 Gradle 的 custom values. 你和各种 Gradle plugin 都可以向 custom value 写入数据. custom value 的数量存在限制. 关于 custom value 目前的最大件数, 请参见 Build scan plugin 文档.
如果你的项目很大, 这些 custom value 的数量可能非常大. 如果超过的限制, 你会在 log 中看到以下信息:
要减少 Kotlin plugin 产生的 custom value 数量, 你可以在 gradle.properties
文件中使用以下属性:
关闭对项目和系统属性的收集
HTTP 构建统计 log 可能包含某些项目和系统属性. 这些属性可以改变构建的行为, 因此将它们输出到构建统计信息中会很有用处. 这是属性也可能存储了敏感信息, 例如, 密码, 或项目的完整路径.
你可以向你的 gradle.properties
文件添加 kotlin.build.report.http.verbose_environment
属性, 来禁止收集这些统计信息.
下一步做什么?
学习: