通过开发构建脚本了解编写 Gradle 的基础知识。

在本部分中,您将

  • 了解 Project 对象

  • 更新构建脚本

  • 更新插件

  • 应用插件

  • 从插件运行任务

步骤 0. 开始之前

  1. 您已在 第 1 部分 中初始化了 Java 应用程序。

  2. 您从 第 2 部分 中了解了 Gradle 构建生命周期。

  3. 您已在 第 3 部分 中添加了子项目和单独的构建。

  4. 您已在 第 4 部分 中查看了设置文件。

步骤 1. Project 对象

构建脚本调用 Gradle API 来配置构建。

在配置阶段,Gradle 会在根目录和子项目目录中查找构建脚本。

当找到构建脚本 build.gradle(.kts) 时,Gradle 会配置 Project 对象。

Project 对象的目的是创建 Task 对象的集合、应用插件和检索依赖项。

您可以在脚本中直接使用 Project 接口上的任何方法和属性。

例如

defaultTasks("some-task")      // Delegates to Project.defaultTasks()
reportsDir = file("reports")   // Delegates to Project.file() and the Java Plugin
defaultTasks 'some-task'      // Delegates to Project.defaultTasks()
reportsDir = file('reports')  // Delegates to Project.file() and the Java Plugin

步骤 2. 构建脚本

让我们分解插件的构建脚本

gradle/license-plugin/plugin/build.gradle.kts
plugins {                                                             (1)
    `java-gradle-plugin`                                              (2)
    id("org.jetbrains.kotlin.jvm") version "1.9.0"                    (3)
}

repositories {                                                        (4)
    mavenCentral()                                                    (5)
}

dependencies {                                                        (6)
    testImplementation("org.jetbrains.kotlin:kotlin-test-junit5")     (7)
    testRuntimeOnly("org.junit.platform:junit-platform-launcher")
}

gradlePlugin {                                                        (8)
    val greeting by plugins.creating {                                (9)
        id = "license.greeting"
        implementationClass = "license.LicensePlugin"
    }
}

// Additional lines //
1 在 Kotlin DSL 中使用 KotlinSettingsScript 中的 plugins{}
2 应用 Java Gradle 插件开发插件以添加对开发 Gradle 插件的支持
3 应用 Kotlin JVM 插件以添加对 Kotlin 的支持
4 使用 Project.repositories() 配置此项目的存储库
5 使用 Maven Central 解析依赖项
6 使用 Project.dependencies() 配置此项目的依赖项
7 使用 Kotlin JUnit 5 集成
8 在 Kotlin DSL 中使用 GradlePluginDevelopmentExtension 中的 gradlePlugin{}
9 定义插件 idimplementationClass
gradle/license-plugin/plugin/build.gradle
plugins {                                                           (1)
    id 'java-gradle-plugin'                                         (2)
    id 'groovy'                                                     (3)
}

repositories {                                                      (4)
    mavenCentral()                                                  (5)
}

dependencies {                                                      (6)
    testImplementation libs.spock.core
    testRuntimeOnly 'org.junit.platform:junit-platform-launcher'
}

gradlePlugin {                                                      (7)
    plugins {
        greeting {
            id = 'license.greeting'                                 (8)
            implementationClass = 'license.LicensePlugin'
        }
    }
}

// Additional lines //
1 在 Groovy DSL 中使用 PluginDependenciesSpec API 中的 plugins{}
2 应用 Java Gradle 插件开发插件以添加对开发 Gradle 插件的支持
3 应用 Groovy 插件以添加对 Groovy 的支持
4 使用 Project.repositories() 配置此项目的存储库
5 使用 Maven Central 解析依赖项
6 使用 Project.dependencies() 配置此项目的依赖项
7 在 Groovy DSL 中使用 PluginAware API 中的 gradlePlugin{}
8 定义插件 idimplementationClass

增强构建功能的插件包含如下

plugins {
    id("java")                          // core plugin, no version required
    id("org.some.plugin") version "2.8" // community plugin, version required
}
plugins {
    id 'java'                          // core plugin, no version required
    id 'org.some.plugin' version '2.8' // community plugin, version required
}

存储库部分让 Gradle 知道从何处提取依赖项

repositories {
    mavenCentral()  // get dependencies from the Maven central repository
}
repositories {
    mavenCentral()  // get dependencies from the Maven central repository
}

依赖项是构建应用程序或库的要求

dependencies {
    // group: 'org.apache.commons', name: 'commons-lang3', version: '3.13.0'
    implementation("org.apache.commons:commons-lang3:3.13.0")
}
dependencies {
    // group: 'org.apache.commons', name: 'commons-lang3', version: '3.13.0'
    implementation 'org.apache.commons:commons-lang3:3.13.0'
}

在此示例中,implementation() 意味着必须将 commons-lang3 库添加到 Java 类路径。

为 Gradle 项目声明的每个依赖项都必须应用于某个范围。也就是说,依赖项在编译时、运行时或两者都需要。这称为配置,当依赖项仅在运行时类路径中需要时,将使用 implementation 配置。

配置块(不要与上述依赖项配置混淆)通常用于配置应用的插件

gradlePlugin {  // Define a custom plugin
    val greeting by plugins.creating {  // Define `greeting` plugin using the `plugins.creating` method
        id = "license.greeting" // Create plugin with the specified ID
        implementationClass = "license.LicensePlugin"   // and specified implementation class
    }
}
gradlePlugin {  // Define a custom plugin
    plugins {
        greeting {  // Define a plugin named greeting
            id = 'license.greeting' // using the id
            implementationClass = 'license.LicensePlugin' // and implementationClass
        }
    }
}

应用 java-gradle-plugin 时,用户必须使用 gradlePlugin{} 配置块配置他们正在开发的插件。

任务是构建期间执行的工作单元。它们可以通过插件或内联定义

val functionalTest by tasks.registering(Test::class) {
    testClassesDirs = functionalTestSourceSet.output.classesDirs
    classpath = functionalTestSourceSet.runtimeClasspath
    useJUnitPlatform()
}

tasks.named<Test>("test") {
    // Use JUnit Jupiter for unit tests.
    useJUnitPlatform()
}
tasks.register('functionalTest', Test) {
    testClassesDirs = sourceSets.functionalTest.output.classesDirs
    classpath = sourceSets.functionalTest.runtimeClasspath
    useJUnitPlatform()
}

tasks.named('test') {
    // Use JUnit Jupiter for unit tests.
    useJUnitPlatform()
}

在 Gradle init 生成的示例中,我们定义了两个任务

  1. functionalTest:此任务使用 tasks.register() 注册。它为功能测试配置测试任务。

  2. test:此任务使用 tasks.named() 为现有的 test 任务配置。它还配置任务以使用 JUnit Jupiter 进行单元测试。

步骤 3. 更新构建脚本

在以下部分中,我们将 LicensePlugin 更新为一个插件,该插件会自动为源代码文件生成许可证标题。我们首先使用新 license 插件的正确名称更新构建脚本

gradle/license-plugin/plugin/build.gradle.kts
gradlePlugin {
    val license by plugins.creating {   // Update name to license
        id = "com.tutorial.license"     // Update id to com.gradle.license
        implementationClass = "license.LicensePlugin"
    }
}
gradle/license-plugin/plugin/build.gradle
gradlePlugin {
    // Define the plugin
    plugins {
        license {                       // Update name to license
            id = 'com.tutorial.license' // Update id to com.gradle.license
            implementationClass = 'license.LicensePlugin'
        }
    }
}

步骤 3. 应用插件

我们把 license 插件应用到 app 子项目

app/build.gradle.kts
plugins {
    application
    id("com.tutorial.license")  // Apply the license plugin
}
app/build.gradle
plugins {
    id 'application'
    id('com.tutorial.license')  // Apply the license plugin
}

步骤 4. 查看插件任务

生成 Gradle 插件项目时,构建 init 会创建一个“hello world”插件。在 LicensePlugin 内部只是一个向控制台打印问候语的任务,任务名称是 greeting

gradle/license-plugin/plugin/src/main/kotlin/license/LicensePlugin.kt
class LicensePlugin: Plugin<Project> {
    override fun apply(project: Project) {                          // Apply plugin
        project.tasks.register("greeting") { task ->                // Register a task
            task.doLast {
                println("Hello from plugin 'com.tutorial.greeting'")  // Hello world printout
            }
        }
    }
}
gradle/license-plugin/plugin/src/main/groovy/license/LicensePlugin.groovy
class LicensePlugin implements Plugin<Project> {
    void apply(Project project) {
        // Register a task
        project.tasks.register("greeting") {
            doLast {
                println("Hello from plugin 'com.tutorial.greeting'")
            }
        }
    }
}

正如我们所见,应用 license 插件时,会公开一个带有简单打印语句的 greeting 任务。

步骤 5. 查看插件任务

license 插件应用到 app 项目时,greeting 任务就会可用

要在根目录中查看任务,请运行

$ ./gradlew tasks --all

------------------------------------------------------------
Tasks runnable from root project 'authoring-tutorial'
------------------------------------------------------------

...

Other tasks
-----------
app:greeting
app:task1
app:task2
lib:task3

最后,使用 ./gradlew greeting 或运行 greeting 任务

$ ./gradlew :app:greeting

> Task :app:greeting
Hello from plugin 'com.tutorial.greeting'

下一步: 编写任务 >>