Skip to content

Kotlin Gradle プラグインにおけるコンパイルとキャッシュ

このページでは、以下のトピックについて学ぶことができます:

増分コンパイル

Kotlin Gradle プラグインは増分コンパイル(incremental compilation)をサポートしており、Kotlin/JVM および Kotlin/JS プロジェクトではデフォルトで有効になっています。 増分コンパイルは、ビルド間でクラスパス内のファイルの変更を追跡し、それらの変更によって影響を受けるファイルのみをコンパイルします。 このアプローチは Gradle のビルドキャッシュと連携し、コンパイル回避(compilation avoidance)をサポートします。

Kotlin/JVM の場合、増分コンパイルはクラスパスのスナップショットに依存しています。 これはモジュールの API 構造をキャプチャし、いつ再コンパイルが必要かを判断します。 パイプライン全体を最適化するために、Kotlin コンパイラは 2 種類のクラスパススナップショットを使用します:

  • 細粒度スナップショット(Fine-grained snapshots): プロパティや関数などのクラスメンバーに関する詳細な情報が含まれます。 メンバーレベルの変更が検出されると、Kotlin コンパイラは変更されたメンバーに依存するクラスのみを再コンパイルします。 パフォーマンスを維持するため、Kotlin Gradle プラグインは Gradle キャッシュ内の .jar ファイルに対しては粗粒度のスナップショットを作成します。
  • 粗粒度スナップショット(Coarse-grained snapshots): クラスの ABI ハッシュのみが含まれます。 ABI の一部が変更されると、Kotlin コンパイラは変更されたクラスに依存するすべてのクラスを再コンパイルします。 これは、外部ライブラリのように頻繁に変更されないクラスに有用です。

Kotlin/JS プロジェクトでは、履歴ファイルに基づいた異なる増分コンパイルアプローチを使用します。

増分コンパイルを無効にするには、いくつかの方法があります:

  • Kotlin/JVM の場合は kotlin.incremental=false を設定します。

  • Kotlin/JS プロジェクトの場合は kotlin.incremental.js=false を設定します。

  • コマンドラインパラメータとして -Pkotlin.incremental=false または -Pkotlin.incremental.js=false を使用します。

    このパラメータは、その後の各ビルドに追加する必要があります。

増分コンパイルを無効にすると、ビルド後に増分キャッシュが無効になります。最初のビルドが増分ビルドになることはありません。

増分コンパイルの問題は、失敗が発生してから数ラウンド後に明らかになることがあります。ビルドレポートを使用して、変更とコンパイルの履歴を追跡してください。これにより、再現可能なバグレポートを提供するのに役立ちます。

現在の増分コンパイルアプローチの仕組みや以前のアプローチとの比較についての詳細は、こちらのブログ記事を参照してください。

Gradle ビルドキャッシュのサポート

Kotlin プラグインは Gradle ビルドキャッシュを使用します。これは、将来のビルドで再利用するためにビルド出力を保存するものです。

すべての Kotlin タスクでキャッシュを無効にするには、システムプロパティ kotlin.caching.enabledfalse に設定します(引数 -Dkotlin.caching.enabled=false を付けてビルドを実行します)。

Gradle コンフィギュレーションキャッシュのサポート

Kotlin プラグインは Gradle コンフィギュレーションキャッシュを使用します。 これにより、以降のビルドでコンフィギュレーションフェーズの結果を再利用することで、ビルドプロセスを高速化します。

コンフィギュレーションキャッシュを有効にする方法については、Gradle のドキュメントを参照してください。この機能を有効にすると、Kotlin Gradle プラグインは自動的にそれを使用し始めます。

Kotlin デーモンと Gradle での使用方法

Kotlin デーモンは以下の通り動作します:

  • プロジェクトをコンパイルするために、Gradle デーモンと共に実行されます。
  • IntelliJ IDEA の内蔵ビルドシステムでプロジェクトをコンパイルする場合は、Gradle デーモンとは別に実行されます。

Kotlin デーモンは、Kotlin コンパイルタスクのいずれかがソースのコンパイルを開始した際、Gradle の実行ステージで起動します。 Kotlin デーモンは、Gradle デーモンが停止したとき、または Kotlin のコンパイルが行われないまま 2 時間経過した後に停止します。

Kotlin デーモンは、Gradle デーモンと同じ JDK を使用します。

Kotlin デーモンの JVM 引数の設定

引数を設定する以下の各方法は、それより前に設定されたものを上書きします:

Gradle デーモン引数の継承

デフォルトでは、Kotlin デーモンは Gradle デーモンから特定の引数セットを継承しますが、Kotlin デーモンに直接指定された JVM 引数がある場合はそれで上書きします。例えば、gradle.properties ファイルに以下の JVM 引数を追加した場合:

none
org.gradle.jvmargs=-Xmx1500m -Xms500m -XX:MaxMetaspaceSize=1g

これらの引数は、Kotlin デーモンの JVM 引数に追加されます:

none
-Xmx1500m -XX:ReservedCodeCacheSize=320m -XX:MaxMetaspaceSize=1g -XX:UseParallelGC -ea -XX:+UseCodeCacheFlushing -XX:+HeapDumpOnOutOfMemoryError -Djava.awt.headless=true -Djava.rmi.server.hostname=127.0.0.1 --add-exports=java.base/sun.nio.ch=ALL-UNNAMED

Kotlin デーモンの JVM 引数に関するデフォルトの動作についての詳細は、Kotlin デーモンの JVM 引数に関する動作を参照してください。

kotlin.daemon.jvm.options システムプロパティ

Gradle デーモンの JVM 引数に kotlin.daemon.jvm.options システムプロパティがある場合は、gradle.properties ファイルでそれを使用します:

none
org.gradle.jvmargs=-Dkotlin.daemon.jvm.options=-Xmx1500m,Xms500m

引数を渡す際は、以下のルールに従ってください:

  • 引数 XmxXX:MaxMetaspaceSizeXX:ReservedCodeCacheSize の前にのみマイナス記号 - を使用してください。
  • 引数はカンマ(,)で区切り、スペースは入れないでください。スペースの後に続く引数は、Kotlin デーモンではなく Gradle デーモン用として扱われます。

以下の条件がすべて満たされる場合、Gradle はこれらのプロパティを無視します:

  • Gradle が JDK 1.9 以上を使用している。
  • Gradle のバージョンが 7.0 から 7.1.1(両端を含む)の間である。
  • Gradle が Kotlin DSL スクリプトをコンパイルしている。
  • Kotlin デーモンが実行されていない。

これを回避するには、Gradle をバージョン 7.2(またはそれ以上)にアップグレードするか、次のセクションで説明する kotlin.daemon.jvmargs プロパティを使用してください。

kotlin.daemon.jvmargs プロパティ

gradle.properties ファイルに kotlin.daemon.jvmargs プロパティを追加できます:

none
kotlin.daemon.jvmargs=-Xmx1500m -Xms500m

ここで、または Gradle の JVM 引数で ReservedCodeCacheSize 引数を指定しない場合、Kotlin Gradle プラグインはデフォルト値の 320m を適用することに注意してください:

none
-Xmx1500m -XX:ReservedCodeCacheSize=320m -Xms500m

kotlin 拡張

kotlin 拡張で引数を指定できます:

kotlin
kotlin {
    kotlinDaemonJvmArgs = listOf("-Xmx486m", "-Xms256m", "-XX:+UseParallelGC")
}
groovy
kotlin {
    kotlinDaemonJvmArgs = ["-Xmx486m", "-Xms256m", "-XX:+UseParallelGC"]
}

特定のタスク定義

特定のタスクに対して引数を指定できます:

kotlin
tasks.withType<CompileUsingKotlinDaemon>().configureEach {
    kotlinDaemonJvmArguments.set(listOf("-Xmx486m", "-Xms256m", "-XX:+UseParallelGC"))
}
groovy
tasks.withType(CompileUsingKotlinDaemon).configureEach { task ->
    task.kotlinDaemonJvmArguments = ["-Xmx1g", "-Xms512m"]
}

この場合、タスクの実行時に新しい Kotlin デーモンインスタンスが起動する可能性があります。詳細は Kotlin デーモンの JVM 引数に関する動作を参照してください。

Kotlin デーモンの JVM 引数に関する動作

Kotlin デーモンの JVM 引数を設定する際は、以下の点に注意してください:

  • サブプロジェクトやタスクごとに異なる JVM 引数セットが指定されている場合、複数の Kotlin デーモンインスタンスが同時に実行されることが想定されています。
  • 新しい Kotlin デーモンインスタンスは、Gradle が関連するコンパイルタスクを実行し、かつ既存の Kotlin デーモンの中に同じ JVM 引数セットを持つものがない場合にのみ起動します。 プロジェクトに多くのサブプロジェクトがあると想像してください。そのほとんどは Kotlin デーモンに一定のヒープメモリを必要としますが、ある一つのモジュールだけが非常に多くのメモリを必要とする(ただし、めったにコンパイルされない)とします。 この場合、そのモジュールに対して異なる JVM 引数セットを提供することで、大きなヒープサイズを持つ Kotlin デーモンを、その特定のモジュールを触る開発者の環境でのみ起動させることができます。

    すでに実行中の Kotlin デーモンがコンパイルリクエストを処理するのに十分なヒープサイズを持っている場合、他の要求された JVM 引数が異なっていても、新しいものを起動する代わりにそのデーモンが再利用されます。

以下の引数が指定されていない場合、Kotlin デーモンは Gradle デーモンからそれらを継承します:

  • -Xmx
  • -XX:MaxMetaspaceSize
  • -XX:ReservedCodeCacheSize。指定も継承もされない場合、デフォルト値は 320m です。

Kotlin デーモンには、以下のデフォルト JVM 引数があります:

  • -XX:UseParallelGC。この引数は、他のガベージコレクタが指定されていない場合にのみ適用されます。
  • -ea
  • -XX:+UseCodeCacheFlushing
  • -Djava.awt.headless=true
  • -D{java.servername.property}={localhostip}
  • --add-exports=java.base/sun.nio.ch=ALL-UNNAMED。この引数は JDK バージョン 16 以上の場合にのみ適用されます。

Kotlin デーモンのデフォルト JVM 引数のリストは、バージョンによって異なる場合があります。VisualVM のようなツールを使用して、Kotlin デーモンのような実行中の JVM プロセスの実際の設定を確認できます。

以前のコンパイラへのロールバック

Kotlin 2.0.0 以降、K2 コンパイラがデフォルトで使用されます。

Kotlin 2.0.0 以降で以前のコンパイラを使用するには、以下のいずれかを行います:

K2 コンパイラの利点についての詳細は、K2 コンパイラ移行ガイドを参照してください。

最新の言語バージョンの試行

Kotlin 2.0.0 以降、最新の言語バージョンを試すには、gradle.properties ファイルで kotlin.experimental.tryNext プロパティを設定してください。 このプロパティを使用すると、Kotlin Gradle プラグインは言語バージョンを、お使いの Kotlin バージョンのデフォルト値より一つ上のものに引き上げます。 例えば、Kotlin 2.0.0 ではデフォルトの言語バージョンは 2.0 なので、このプロパティは言語バージョン 2.1 を設定します。

あるいは、以下のコマンドを実行することもできます:

shell
./gradlew assemble -Pkotlin.experimental.tryNext=true

ビルドレポートで、各タスクのコンパイルに使用された言語バージョンを確認できます。

ビルドレポート

ビルドレポートには、さまざまなコンパイルフェーズの所要時間や、コンパイルが増分にならなかった理由が含まれます。 コンパイル時間が長すぎる場合や、同じプロジェクトでも時間が異なる場合など、パフォーマンスの問題を調査するためにビルドレポートを使用してください。

Kotlin ビルドレポートは、単一の Gradle タスクを最小単位とする Gradle Build Scans よりも効率的に、ビルドパフォーマンスの問題を調査するのに役立ちます。

実行時間の長いコンパイルにおいて、ビルドレポートを分析することで解決できる一般的なケースが 2 つあります:

  • ビルドが増分ではなかった。理由を分析し、根本的な問題を修正してください。
  • ビルドは増分だったが、時間がかかりすぎた。ソースファイルの再編成(大きなファイルの分割、個別のクラスを異なるファイルに保存、巨大なクラスのリファクタリング、トップレベル関数を異なるファイルで宣言するなど)を試みてください。

ビルドレポートには、プロジェクトで使用されている Kotlin バージョンも表示されます。さらに、Kotlin 1.9.0 以降では、Gradle Build Scans でコードのコンパイルにどのコンパイラが使用されたかを確認できるようになりました。

ビルドレポートの読み方および JetBrains におけるビルドレポートの活用方法について詳細を確認してください。

ビルドレポートの有効化

ビルドレポートを有効にするには、gradle.properties でビルドレポートの出力先を宣言します:

none
kotlin.build.report.output=file

出力先には、以下の値とその組み合わせが使用可能です:

オプション説明
fileビルドレポートを人間が読みやすい形式でローカルファイルに保存します。デフォルトでは ${project_folder}/build/reports/kotlin-build/${project_name}-timestamp.txt です。
single_fileビルドレポートをオブジェクトの形式で指定されたローカルファイルに保存します。
build_scanビルドレポートを ビルドスキャンcustom values セクションに保存します。Gradle Enterprise プラグインは、カスタム値の数とその長さを制限していることに注意してください。大規模なプロジェクトでは、一部の値が失われる可能性があります。
httpHTTP(S) を使用してビルドレポートを投稿します。POST メソッドでメトリクスを JSON 形式で送信します。送信データの現在のバージョンは Kotlin リポジトリで確認できます。HTTP エンドポイントのサンプルは、こちらのブログ記事にあります。
jsonビルドレポートを JSON 形式でローカルファイルに保存します。ビルドレポートの場所は kotlin.build.report.json.directory で設定します(下記参照)。デフォルトの名前は ${project_name}-build-<date-time>-<index>.json です。

kotlin.build.report で使用可能なオプションのリストは以下の通りです:

none
# 必須の出力先。任意の組み合わせが可能です
kotlin.build.report.output=file,single_file,http,build_scan,json

# single_file 出力を使用する場合は必須。レポートの保存場所
# 非推奨の `kotlin.internal.single.build.metrics.file` プロパティの代わりに使用してください
kotlin.build.report.single_file=some_filename

# json 出力を使用する場合は必須。レポートの保存場所
kotlin.build.report.json.directory=my/directory/path

# オプション。ファイルベースのレポートの出力ディレクトリ。デフォルト:build/reports/kotlin-build/
kotlin.build.report.file.output_dir=kotlin-reports

# オプション。ビルドレポートを識別するためのラベル(例:デバッグパラメータなど)
kotlin.build.report.label=some_label

HTTP にのみ適用されるオプション:

none
# 必須。HTTP(S) ベースのレポートの投稿先
kotlin.build.report.http.url=http://127.0.0.1:8080

# オプション。HTTP エンドポイントが認証を必要とする場合のユーザー名とパスワード
kotlin.build.report.http.user=someUser
kotlin.build.report.http.password=somePassword

# オプション。ビルドレポートにビルドの Git ブランチ名を追加する
kotlin.build.report.http.include_git_branch.name=true|false

# オプション。ビルドレポートにコンパイラ引数を追加する
# プロジェクトに多くのモジュールが含まれている場合、レポート内のコンパイラ引数が非常に重くなり、あまり役立たない場合があります
kotlin.build.report.include_compiler_arguments=true|false

カスタム値の制限

ビルドスキャンの統計情報を収集するために、Kotlin ビルドレポートは Gradle のカスタム値を使用します。 ユーザー自身と、さまざまな Gradle プラグインの両方がカスタム値にデータを書き込むことができます。カスタム値の数には制限があります。 現在の最大カスタム値数は、Build scan プラグインのドキュメントで確認してください。

大規模なプロジェクトの場合、このようなカスタム値の数が非常に多くなることがあります。この数が制限を超えると、ログに以下のメッセージが表示されることがあります:

text
Maximum number of custom values (1,000) exceeded

Kotlin プラグインが生成するカスタム値の数を減らすには、gradle.properties で以下のプロパティを使用できます:

none
kotlin.build.report.build_scan.custom_values_limit=500

プロジェクトおよびシステムプロパティの収集の停止

HTTP ビルド統計ログには、一部のプロジェクトおよびシステムプロパティが含まれる場合があります。これらのプロパティはビルドの動作を変更する可能性があるため、ビルド統計にログを記録しておくと便利です。 ただし、これらのプロパティにはパスワードやプロジェクトのフルパスなどの機密データが含まれている可能性があります。

gradle.propertieskotlin.build.report.http.verbose_environment プロパティを追加することで、これらの統計情報の収集を無効にできます。

JetBrains はこれらの統計情報を収集しません。ユーザーがレポートの保存場所を選択します。

次に学ぶこと

以下について詳細を確認してください: