學習使用子專案和複合建置來構建 Gradle 專案的基本知識。
步驟 1. 關於多專案建置
通常,建置包含多個專案,例如將部署在您的生態系統中的共享函式庫或獨立應用程式。
在 Gradle 中,多專案建置包含:
-
settings.gradle(.kts)
檔案,代表您的 Gradle 建置,包括所需的子專案,例如 include("app", "model", "service") -
每個子專案在相應子目錄中的
build.gradle(.kts)
和原始碼
我們的建置目前包含一個名為 authoring-tutorial
的根專案,其中有一個 app
子專案
. (1)
├── app (2)
│ ... (3)
│ └── build.gradle.kts (4)
└── settings.gradle.kts (5)
1 | authoring-tutorial 根專案 |
2 | app 子專案 |
3 | app 原始碼 |
4 | app 建置腳本 |
5 | 可選的設定檔 |
. (1)
├── app (2)
│ ... (3)
│ └── build.gradle (4)
└── settings.gradle (5)
1 | authoring-tutorial 根專案 |
2 | app 子專案 |
3 | app 原始碼 |
4 | app 建置腳本 |
5 | 可選的設定檔 |
步驟 2. 新增另一個子專案到建置
想像一下,我們的專案正在成長,並且需要一個自訂函式庫才能運作。
讓我們建立這個虛構的 lib
。首先,建立一個 lib
資料夾
mkdir lib
cd lib
建立一個名為 build.gradle(.kts)
的檔案,並將以下行添加到其中
plugins {
id("java")
}
repositories {
mavenCentral()
}
dependencies {
testImplementation("org.junit.jupiter:junit-jupiter:5.9.3")
testRuntimeOnly("org.junit.platform:junit-platform-launcher")
implementation("com.google.guava:guava:32.1.1-jre")
}
tasks.named<Test>("test") {
useJUnitPlatform()
}
tasks.register("task3"){
println("REGISTER TASK3: This is executed during the configuration phase")
}
tasks.named("task3"){
println("NAMED TASK3: This is executed during the configuration phase")
doFirst {
println("NAMED TASK3 - doFirst: This is executed during the execution phase")
}
doLast {
println("NAMED TASK3 - doLast: This is executed during the execution phase")
}
}
plugins {
id 'java'
}
repositories {
mavenCentral()
}
dependencies {
testImplementation 'org.junit.jupiter:junit-jupiter:5.9.3'
testRuntimeOnly 'org.junit.platform:junit-platform-launcher'
implementation 'com.google.guava:guava:32.1.1-jre'
}
test {
useJUnitPlatform()
}
tasks.register('task3') {
println('REGISTER TASK3: This is executed during the configuration phase')
}
tasks.named('task3') {
println('NAMED TASK3: This is executed during the configuration phase')
doFirst {
println('NAMED TASK3 - doFirst: This is executed during the execution phase')
}
doLast {
println('NAMED TASK3 - doLast: This is executed during the execution phase')
}
}
您的專案應該看起來像這樣
.
├── app
│ ...
│ └── build.gradle.kts
├── lib
│ └── build.gradle.kts
└── settings.gradle.kts
.
├── app
│ ...
│ └── build.gradle
├── lib
│ └── build.gradle
└── settings.gradle
讓我們將一些程式碼添加到 lib
子專案。建立一個新目錄
mkdir -p lib/src/main/java/com/gradle
在名為 CustomLib.java
的檔案中建立一個名為 CustomLib
的 Java 類別,並包含以下原始碼
package com.gradle;
public class CustomLib {
public static String identifier = "I'm a String from a lib.";
}
專案現在應該具有以下檔案和目錄結構
.
├── app
│ ├── build.gradle.kts
│ └── src
│ └── main
│ └── java
│ └── authoring
│ └── tutorial
│ └── App.java
├── lib
│ ├── build.gradle.kts
│ └── src
│ └── main
│ └── java
│ └── com
│ └── gradle
│ └── CustomLib.java
└── settings.gradle.kts
.
├── app
│ ├── build.gradle
│ └── src
│ └── main
│ └── java
│ └── authoring
│ └── tutorial
│ └── App.java
├── lib
│ ├── build.gradle
│ └── src
│ └── main
│ └── java
│ └── com
│ └── gradle
│ └── CustomLib.java
└── settings.gradle
但是,lib
子專案不屬於建置,並且您將無法執行 task3
,直到將其添加到 settings.gradle(.kts)
檔案中。
若要將 lib
新增到建置,請相應地更新根目錄中的 settings.gradle(.kts)
檔案。
plugins {
id("org.gradle.toolchains.foojay-resolver-convention") version "0.9.0"
}
rootProject.name = "authoring-tutorial"
include("app")
include("lib") // Add lib to the build
plugins {
id 'org.gradle.toolchains.foojay-resolver-convention' version '0.9.0'
}
rootProject.name = 'authoring-tutorial'
include('app')
include('lib') // Add lib to the build
讓我們在 app/build.gradle(.kts)
中將 lib
子專案新增為 app
相依性。
dependencies {
implementation(project(":lib")) // Add lib as an app dependency
}
dependencies {
implementation(project(':lib')) // Add lib as an app dependency
}
更新 app
原始碼,使其匯入 lib
package authoring.tutorial;
import com.gradle.CustomLib;
public class App {
public String getGreeting() {
return "CustomLib identifier is: " + CustomLib.identifier;
}
public static void main(String[] args) {
System.out.println(new App().getGreeting());
}
}
最後,讓我們使用命令 ./gradlew run
執行 app
$ ./gradlew run
> Configure project :app
> Task :app:processResources NO-SOURCE
> Task :lib:compileJava
> Task :lib:processResources NO-SOURCE
> Task :lib:classes
> Task :lib:jar
> Task :app:compileJava
> Task :app:classes
> Task :app:run
CustomLib identifier is: I'm a String from a lib.
BUILD SUCCESSFUL in 11s
8 actionable tasks: 6 executed, 2 up-to-date
我們根專案 authoring-tutorial
的建置現在包含兩個子專案,app
和 lib
。app
依賴於 lib
。您可以獨立於 app
建置 lib
。但是,若要建置 app
,Gradle 也會建置 lib
。
步驟 3. 了解複合建置
複合建置只是一個包含其他建置的建置。
複合建置允許您:
-
從專案建置中提取您的建置邏輯(並在子專案之間重複使用)
-
組合通常獨立開發的建置(例如外掛程式和應用程式)
-
將大型建置分解為更小、更隔離的區塊
步驟 4. 新增建置到建置
讓我們將外掛程式添加到我們的建置中。首先,在 gradle
目錄中建立一個名為 license-plugin
的新目錄。
cd gradle
mkdir license-plugin
cd license-plugin
一旦進入 gradle/license-plugin
目錄,執行 gradle init
。請確保您選擇 Gradle plugin
專案以及以下 init
任務的其他選項。
$ gradle init --dsl kotlin --type kotlin-gradle-plugin --project-name license
$ gradle init --dsl groovy --type groovy-gradle-plugin --project-name license
為任何其他提示選擇預設值。
您的專案應該看起來像這樣
.
├── app
│ ...
│ └── build.gradle.kts
├── lib
│ ...
│ └── build.gradle.kts
├── gradle
│ ├── ...
│ └── license-plugin
│ ├── settings.gradle.kts
│ └── plugin
│ ├── gradle
│ │ └── ....
│ ├── src
│ │ ├── functionalTest
│ │ │ └── ....
│ │ ├── main
│ │ │ └── kotlin
│ │ │ └── license
│ │ │ └── LicensePlugin.kt
│ │ └── test
│ │ └── ...
│ └── build.gradle.kts
│
└── settings.gradle.kts
.
├── app
│ ...
│ └── build.gradle
├── lib
│ ...
│ └── build.gradle
├── gradle
│ ├── ...
│ └── license-plugin
│ ├── settings.gradle
│ └── plugin
│ ├── gradle
│ │ └── ....
│ ├── src
│ │ ├── functionalTest
│ │ │ └── ....
│ │ ├── main
│ │ │ └── groovy
│ │ │ └── license
│ │ │ └── LicensePlugin.groovy
│ │ └── test
│ │ └── ...
│ └── build.gradle
│
└── settings.gradle
花時間查看 LicensePlugin.kt
或 LicensePlugin.groovy
程式碼以及 gradle/license-plugin/settings.gradle(.kts)
檔案。重要的是要注意,這是一個完全獨立的建置,具有自己的設定檔和建置腳本。
rootProject.name = "license"
include("plugin")
rootProject.name = 'license'
include('plugin')
若要將我們的 license-plugin
建置新增到根專案,請相應地更新根目錄 settings.gradle(.kts)
檔案。
plugins {
id("org.gradle.toolchains.foojay-resolver-convention") version "0.9.0"
}
rootProject.name = "authoring-tutorial"
include("app")
include("subproject")
includeBuild("gradle/license-plugin") // Add the new build
plugins {
id 'org.gradle.toolchains.foojay-resolver-convention' version '0.9.0'
}
rootProject.name = 'running-tutorial-groovy'
include('app')
include('lib')
includeBuild('gradle/license-plugin')
您可以通過在根資料夾 authoring-tutorial
中執行 ./gradlew projects
來查看根專案的結構。
$ ./gradlew projects
------------------------------------------------------------
Root project 'authoring-tutorial'
------------------------------------------------------------
Root project 'authoring-tutorial'
+--- Project ':app'
\--- Project ':lib'
Included builds
\--- Included build ':license-plugin'
我們根專案 authoring-tutorial
的建置現在包含兩個子專案,app
和 lib
,以及另一個建置,license-plugin
。
當在專案根目錄中執行時:
-
./gradlew build
- 建置app
和lib
-
./gradlew :app:build
- 建置app
和lib
-
./gradlew :lib:build
- 僅建置lib
-
./gradlew :license-plugin:plugin:build
- 僅建置license-plugin
有許多方法可以使用 Gradle 設計專案的架構。
多專案建置非常適合組織具有許多模組的專案,例如 mobile-app
、web-app
、api
、lib
和 documentation
,它們之間具有相依性。
複合(包含)建置非常適合分離建置邏輯(即,慣例外掛程式)或測試系統(即,修補函式庫)
下一步: 設定檔 >>