任务表示构建执行的独立工作单元,例如编译类、创建 JAR、生成 Javadoc 或将归档发布到仓库。

writing tasks 1

在阅读本章之前,建议您先阅读学习基础知识并完成教程

列出任务

项目中的所有可用任务都来自 Gradle 插件和构建脚本。

您可以通过在终端中运行以下命令来列出项目中的所有可用任务

$ ./gradlew tasks

让我们以一个非常基本的 Gradle 项目为例。该项目具有以下结构

gradle-project
├── app
│   ├── build.gradle.kts    // empty file - no build logic
│   └── ...                 // some java code
├── settings.gradle.kts     // includes app subproject
├── gradle
│   └── ...
├── gradlew
└── gradlew.bat
gradle-project
├── app
│   ├── build.gradle    // empty file - no build logic
│   └── ...             // some java code
├── settings.gradle     // includes app subproject
├── gradle
│   └── ...
├── gradlew
└── gradlew.bat

settings 文件包含以下内容

settings.gradle.kts
rootProject.name = "gradle-project"
include("app")
settings.gradle
rootProject.name = 'gradle-project'
include('app')

目前,app 子项目的构建文件为空。

要查看 app 子项目中可用的任务,请运行 ./gradlew :app:tasks

$ ./gradlew :app:tasks

> Task :app:tasks

------------------------------------------------------------
Tasks runnable from project ':app'
------------------------------------------------------------

Help tasks
----------
buildEnvironment - Displays all buildscript dependencies declared in project ':app'.
dependencies - Displays all dependencies declared in project ':app'.
dependencyInsight - Displays the insight into a specific dependency in project ':app'.
help - Displays a help message.
javaToolchains - Displays the detected java toolchains.
kotlinDslAccessorsReport - Prints the Kotlin code for accessing the currently available project extensions and conventions.
outgoingVariants - Displays the outgoing variants of project ':app'.
projects - Displays the sub-projects of project ':app'.
properties - Displays the properties of project ':app'.
resolvableConfigurations - Displays the configurations that can be resolved in project ':app'.
tasks - Displays the tasks runnable from project ':app'.

我们观察到目前只有少量帮助任务可用。这是因为 Gradle 的核心仅提供分析构建的任务。其他任务(例如构建项目或编译代码的任务)由插件添加。

让我们通过将 Gradle 核心 base 插件 添加到 app 构建脚本来探索这一点

app/build.gradle.kts
plugins {
    id("base")
}
app/build.gradle
plugins {
    id('base')
}

base 插件添加了中心生命周期任务。现在,当我们运行 ./gradlew app:tasks 时,我们可以看到 assemblebuild 任务可用

$ ./gradlew :app:tasks

> Task :app:tasks

------------------------------------------------------------
Tasks runnable from project ':app'
------------------------------------------------------------

Build tasks
-----------
assemble - Assembles the outputs of this project.
build - Assembles and tests this project.
clean - Deletes the build directory.

Help tasks
----------
buildEnvironment - Displays all buildscript dependencies declared in project ':app'.
dependencies - Displays all dependencies declared in project ':app'.
dependencyInsight - Displays the insight into a specific dependency in project ':app'.
help - Displays a help message.
javaToolchains - Displays the detected java toolchains.
outgoingVariants - Displays the outgoing variants of project ':app'.
projects - Displays the sub-projects of project ':app'.
properties - Displays the properties of project ':app'.
resolvableConfigurations - Displays the configurations that can be resolved in project ':app'.
tasks - Displays the tasks runnable from project ':app'.

Verification tasks
------------------
check - Runs all checks.

任务结果

当 Gradle 执行任务时,它会通过控制台使用结果标记任务。

author tasks 1

这些标签基于任务是否具有要执行的操作以及 Gradle 是否执行了这些操作。操作包括但不限于编译代码、压缩文件和发布归档。

(无标签)已执行

任务执行了其操作。

  • 任务具有操作,并且 Gradle 执行了这些操作。

  • 任务没有操作和一些依赖项,并且 Gradle 执行了一个或多个依赖项。另请参阅生命周期任务

最新

任务的输出未更改。

  • 任务具有输出和输入,但它们没有更改。请参阅增量构建

  • 任务具有操作,但任务告诉 Gradle 它没有更改其输出。

  • 任务没有操作和一些依赖项,但所有依赖项都是 最新已跳过来自缓存。请参阅生命周期任务

  • 任务没有操作,也没有依赖项。

来自缓存

可以从之前的执行中找到任务的输出。

  • 任务具有从构建缓存还原的输出。请参阅构建缓存

已跳过

任务未执行其操作。

无源

任务不需要执行其操作。

  • 任务具有输入和输出,但没有源(即,未找到输入)。

任务组和描述

任务组和描述用于组织和描述任务。

任务组用于对任务进行分类。当您运行 ./gradlew tasks 时,任务会列在其各自的组下,从而更容易理解其目的以及与其他任务的关系。组使用 group 属性设置。

描述

描述简要说明了任务的作用。当您运行 ./gradlew tasks 时,描述会显示在每个任务旁边,帮助您了解其目的以及如何使用它。描述使用 description 属性设置。

让我们以一个基本的 Java 应用程序为例。构建包含一个名为 app 的子项目。

让我们列出目前 app 中可用的任务

$ ./gradlew :app:tasks

> Task :app:tasks

------------------------------------------------------------
Tasks runnable from project ':app'
------------------------------------------------------------

Application tasks
-----------------
run - Runs this project as a JVM application.

Build tasks
-----------
assemble - Assembles the outputs of this project.

在这里,:run 任务是 Application 组的一部分,描述为 将此项目作为 JVM 应用程序运行。在代码中,它看起来像这样

app/build.gradle.kts
tasks.register("run") {
    group = "Application"
    description = "Runs this project as a JVM application."
}
app/build.gradle
tasks.register("run") {
    group = "Application"
    description = "Runs this project as a JVM application."
}

私有和隐藏任务

Gradle 不支持将任务标记为私有

但是,仅当设置了 task.group 或没有其他任务依赖于它时,任务才会运行 :tasks 时显示。

例如,以下任务在运行 ./gradlew :app:tasks 时不会出现,因为它没有组;它被称为隐藏任务

app/build.gradle.kts
tasks.register("helloTask") {
    println("Hello")
}
app/build.gradle
tasks.register("helloTask") {
    println 'Hello'
}

尽管 helloTask 未列出,但它仍然可以由 Gradle 执行

$ ./gradlew :app:tasks

> Task :app:tasks

------------------------------------------------------------
Tasks runnable from project ':app'
------------------------------------------------------------

Application tasks
-----------------
run - Runs this project as a JVM application

Build tasks
-----------
assemble - Assembles the outputs of this project.

让我们向同一任务添加一个组

app/build.gradle.kts
tasks.register("helloTask") {
    group = "Other"
    description = "Hello task"
    println("Hello")
}
app/build.gradle
tasks.register("helloTask") {
    group = "Other"
    description = "Hello task"
    println 'Hello'
}

现在添加了组,该任务可见

$ ./gradlew :app:tasks

> Task :app:tasks

------------------------------------------------------------
Tasks runnable from project ':app'
------------------------------------------------------------

Application tasks
-----------------
run - Runs this project as a JVM application

Build tasks
-----------
assemble - Assembles the outputs of this project.

Other tasks
-----------
helloTask - Hello task

相反,./gradlew tasks --all 将显示所有任务;隐藏可见任务都将被列出。

分组任务

如果您想自定义列出时向用户显示的任务,您可以对任务进行分组并设置每个组的可见性。

请记住,即使您隐藏任务,它们仍然可用,Gradle 仍然可以运行它们。

让我们从 Gradle init 为具有多个子项目的 Java 应用程序构建的示例开始。项目结构如下

gradle-project
├── app
│   ├── build.gradle.kts
│   └── src                 // some java code
│       └── ...
├── utilities
│   ├── build.gradle.kts
│   └── src                 // some java code
│       └── ...
├── list
│   ├── build.gradle.kts
│   └── src                 // some java code
│       └── ...
├── buildSrc
│   ├── build.gradle.kts
│   ├── settings.gradle.kts
│   └── src                 // common build logic
│       └── ...
├── settings.gradle.kts
├── gradle
├── gradlew
└── gradlew.bat
gradle-project
├── app
│   ├── build.gradle
│   └── src             // some java code
│       └── ...
├── utilities
│   ├── build.gradle
│   └── src             // some java code
│       └── ...
├── list
│   ├── build.gradle
│   └── src             // some java code
│       └── ...
├── buildSrc
│   ├── build.gradle
│   ├── settings.gradle
│   └── src             // common build logic
│       └── ...
├── settings.gradle
├── gradle
├── gradlew
└── gradlew.bat

运行 app:tasks 以查看 app 子项目中可用的任务

$ ./gradlew :app:tasks

> Task :app:tasks

------------------------------------------------------------
Tasks runnable from project ':app'
------------------------------------------------------------

Application tasks
-----------------
run - Runs this project as a JVM application

Build tasks
-----------
assemble - Assembles the outputs of this project.
build - Assembles and tests this project.
buildDependents - Assembles and tests this project and all projects that depend on it.
buildNeeded - Assembles and tests this project and all projects it depends on.
classes - Assembles main classes.
clean - Deletes the build directory.
jar - Assembles a jar archive containing the classes of the 'main' feature.
testClasses - Assembles test classes.

Distribution tasks
------------------
assembleDist - Assembles the main distributions
distTar - Bundles the project as a distribution.
distZip - Bundles the project as a distribution.
installDist - Installs the project as a distribution as-is.

Documentation tasks
-------------------
javadoc - Generates Javadoc API documentation for the 'main' feature.

Help tasks
----------
buildEnvironment - Displays all buildscript dependencies declared in project ':app'.
dependencies - Displays all dependencies declared in project ':app'.
dependencyInsight - Displays the insight into a specific dependency in project ':app'.
help - Displays a help message.
javaToolchains - Displays the detected java toolchains.
kotlinDslAccessorsReport - Prints the Kotlin code for accessing the currently available project extensions and conventions.
outgoingVariants - Displays the outgoing variants of project ':app'.
projects - Displays the sub-projects of project ':app'.
properties - Displays the properties of project ':app'.
resolvableConfigurations - Displays the configurations that can be resolved in project ':app'.
tasks - Displays the tasks runnable from project ':app'.

Verification tasks
------------------
check - Runs all checks.
test - Runs the test suite.

如果我们查看可用任务列表,即使对于标准的 Java 项目,它也是很广泛的。开发人员在使用构建时很少直接需要这些任务中的许多任务。

我们可以配置 :tasks 任务并将显示的任务限制为特定组。

让我们创建我们自己的组,以便默认情况下隐藏所有任务,方法是更新 app 构建脚本

app/build.gradle.kts
val myBuildGroup = "my app build"               // Create a group name

tasks.register<TaskReportTask>("tasksAll") {    // Register the tasksAll task
group = myBuildGroup
description = "Show additional tasks."
setShowDetail(true)
}

tasks.named<TaskReportTask>("tasks") {          // Move all existing tasks to the group
displayGroup = myBuildGroup
}
app/build.gradle
def myBuildGroup = "my app build"               // Create a group name

tasks.register("tasksAll", TaskReportTask) {    // Register the tasksAll task
    group = myBuildGroup
    description = "Show additional tasks."
    setShowDetail(true)
}

tasks.named("tasks", TaskReportTask) {          // Move all existing tasks to the group
    displayGroup = myBuildGroup
}

现在,当我们列出 app 中可用的任务时,列表会更短

$ ./gradlew :app:tasks

> Task :app:tasks

------------------------------------------------------------
Tasks runnable from project ':app'
------------------------------------------------------------

My app build tasks
------------------
tasksAll - Show additional tasks.

任务类别

Gradle 区分两种任务类别

  1. 生命周期任务

  2. 可操作任务

生命周期任务定义您可以调用的目标,例如 :build 您的项目。生命周期任务不向 Gradle 提供操作。它们必须连接到可操作任务。base Gradle 插件 仅添加生命周期任务。

可操作任务定义 Gradle 要执行的操作,例如 :compileJava,它编译项目的 Java 代码。操作包括创建 JAR、压缩文件、发布归档等等。像 java-library 插件 这样的插件会添加可操作任务。

让我们更新先前示例的构建脚本,该脚本当前是一个空文件,以便我们的 app 子项目是一个 Java 库

app/build.gradle.kts
plugins {
    id("java-library")
}
app/build.gradle
plugins {
    id('java-library')
}

再次,我们列出可用的任务,以查看有哪些新任务可用

$ ./gradlew :app:tasks

> Task :app:tasks

------------------------------------------------------------
Tasks runnable from project ':app'
------------------------------------------------------------

Build tasks
-----------
assemble - Assembles the outputs of this project.
build - Assembles and tests this project.
buildDependents - Assembles and tests this project and all projects that depend on it.
buildNeeded - Assembles and tests this project and all projects it depends on.
classes - Assembles main classes.
clean - Deletes the build directory.
jar - Assembles a jar archive containing the classes of the 'main' feature.
testClasses - Assembles test classes.

Documentation tasks
-------------------
javadoc - Generates Javadoc API documentation for the 'main' feature.

Help tasks
----------
buildEnvironment - Displays all buildscript dependencies declared in project ':app'.
dependencies - Displays all dependencies declared in project ':app'.
dependencyInsight - Displays the insight into a specific dependency in project ':app'.
help - Displays a help message.
javaToolchains - Displays the detected java toolchains.
outgoingVariants - Displays the outgoing variants of project ':app'.
projects - Displays the sub-projects of project ':app'.
properties - Displays the properties of project ':app'.
resolvableConfigurations - Displays the configurations that can be resolved in project ':app'.
tasks - Displays the tasks runnable from project ':app'.

Verification tasks
------------------
check - Runs all checks.
test - Runs the test suite.

我们看到许多新任务可用,例如 jartestClasses

此外,java-library 插件已将可操作任务连接到生命周期任务。如果我们调用 :build 任务,我们可以看到已执行了多个任务,包括 :app:compileJava 任务。

$./gradlew :app:build

> Task :app:compileJava
> Task :app:processResources NO-SOURCE
> Task :app:classes
> Task :app:jar
> Task :app:assemble
> Task :app:compileTestJava
> Task :app:processTestResources NO-SOURCE
> Task :app:testClasses
> Task :app:test
> Task :app:check
> Task :app:build

可操作的 :compileJava 任务已连接到生命周期 :build 任务。

增量任务

Gradle 任务的一个关键特性是它们的增量性质。

Gradle 可以重用先前构建的结果。因此,如果我们之前构建过我们的项目并且只做了少量更改,则重新运行 :build 将不需要 Gradle 执行大量工作。

例如,如果我们仅修改项目中的测试代码,而保持生产代码不变,则执行构建将仅重新编译测试代码。Gradle 将生产代码的任务标记为 最新,表示自上次成功构建以来它保持不变

$./gradlew :app:build

gradle@MacBook-Pro temp1 % ./gradlew :app:build
> Task :app:compileJava UP-TO-DATE
> Task :app:processResources NO-SOURCE
> Task :app:classes UP-TO-DATE
> Task :app:jar UP-TO-DATE
> Task :app:assemble UP-TO-DATE
> Task :app:compileTestJava
> Task :app:processTestResources NO-SOURCE
> Task :app:testClasses
> Task :app:test
> Task :app:check UP-TO-DATE
> Task :app:build UP-TO-DATE

缓存任务

Gradle 可以使用构建缓存重用过去构建的结果。

要启用此功能,请使用 --build-cache 命令行 参数或在 gradle.properties 文件中设置 org.gradle.caching=true 来激活构建缓存。

此优化有可能显着加速您的构建

$./gradlew :app:clean :app:build --build-cache

> Task :app:compileJava FROM-CACHE
> Task :app:processResources NO-SOURCE
> Task :app:classes UP-TO-DATE
> Task :app:jar
> Task :app:assemble
> Task :app:compileTestJava FROM-CACHE
> Task :app:processTestResources NO-SOURCE
> Task :app:testClasses UP-TO-DATE
> Task :app:test FROM-CACHE
> Task :app:check UP-TO-DATE
> Task :app:build

当 Gradle 可以从缓存中获取任务的输出时,它会将任务标记为 来自缓存

如果您经常在分支之间切换,则构建缓存非常方便。Gradle 支持本地和远程构建缓存。

开发任务

在开发 Gradle 任务时,您有两种选择

  1. 使用现有的 Gradle 任务类型,例如 ZipCopyDelete

  2. 创建您自己的 Gradle 任务类型,例如 MyResolveTaskCustomTaskUsingToolchains

任务类型只是 Gradle Task 类的子类。

对于 Gradle 任务,需要考虑三种状态

  1. 注册任务 - 在您的构建逻辑中使用任务(由您实现或由 Gradle 提供)。

  2. 配置任务 - 为已注册的任务定义输入和输出。

  3. 实现任务 - 创建自定义任务类(即,自定义类类型)。

注册通常使用 register() 方法完成。
配置任务通常使用 named() 方法完成。
实现任务通常通过扩展 Gradle 的 DefaultTask 类来完成

tasks.register<Copy>("myCopy")                              (1)

tasks.named<Copy>("myCopy") {                               (2)
    from("resources")
    into("target")
    include("**/*.txt", "**/*.xml", "**/*.properties")
}

abstract class MyCopyTask : DefaultTask() {                 (3)
    @TaskAction
    fun copyFiles() {
        val sourceDir = File("sourceDir")
        val destinationDir = File("destinationDir")
        sourceDir.listFiles()?.forEach { file ->
            if (file.isFile && file.extension == "txt") {
                file.copyTo(File(destinationDir, file.name))
            }
        }
    }
}
1 注册 myCopy 类型为 Copy 的任务,以让 Gradle 知道我们打算在构建逻辑中使用它。
2 根据其 API,使用其所需的输入和输出配置已注册的 myCopy 任务。
3 实现一个名为 MyCopyTask 的自定义任务类型,该类型扩展了 DefaultTask 并定义了 copyFiles 任务操作。
tasks.register("myCopy", Copy)                              (1)

tasks.named("myCopy", Copy) {                               (2)
    from "resources"
    into "target"
    include "**/*.txt", "**/*.xml", "**/*.properties"
}

abstract class MyCopyTask extends DefaultTask {             (3)
    @TaskAction
    void copyFiles() {
        fileTree('sourceDir').matching {
            include '**/*.txt'
        }.forEach { file ->
            file.copyTo(file.path.replace('sourceDir', 'destinationDir'))
        }
    }
}
1 注册 myCopy 类型为 Copy 的任务,以让 Gradle 知道我们打算在构建逻辑中使用它。
2 根据其 API,使用其所需的输入和输出配置已注册的 myCopy 任务。
3 实现一个名为 MyCopyTask 的自定义任务类型,该类型扩展了 DefaultTask 并定义了 copyFiles 任务操作。

1. 注册任务

您可以通过在构建脚本或插件中注册任务来定义 Gradle 要执行的操作。

任务是使用字符串定义任务名称的

build.gradle.kts
tasks.register("hello") {
    doLast {
        println("hello")
    }
}
build.gradle
tasks.register('hello') {
    doLast {
        println 'hello'
    }
}

在上面的示例中,任务使用 register() 方法在 TaskContainer 中添加到 TasksCollection

2. 配置任务

必须配置 Gradle 任务才能成功完成其操作。如果任务需要 ZIP 文件,则必须配置文件名和位置。您可以参考 Gradle Zip 任务的 API 来了解如何正确配置它。

让我们看一下 Gradle 提供的 Copy 任务示例。我们首先在构建脚本中注册一个名为 myCopy 类型为 Copy 的任务

build.gradle.kts
tasks.register<Copy>("myCopy")
build.gradle
tasks.register('myCopy', Copy)

这将注册一个没有默认行为的复制任务。由于该任务的类型为 Copy(Gradle 支持的任务类型),因此可以使用其 API 进行配置。

以下示例显示了实现相同配置的几种方法

1. 使用 named() 方法:

使用 named() 配置在其他位置注册的现有任务

build.gradle.kts
tasks.named<Copy>("myCopy") {
    from("resources")
    into("target")
    include("**/*.txt", "**/*.xml", "**/*.properties")
}
build.gradle
tasks.named('myCopy') {
    from 'resources'
    into 'target'
    include('**/*.txt', '**/*.xml', '**/*.properties')
}

2. 使用配置块:

使用块在注册任务后立即配置任务

build.gradle.kts
tasks.register<Copy>("copy") {
   from("resources")
   into("target")
   include("**/*.txt", "**/*.xml", "**/*.properties")
}
build.gradle
tasks.register('copy', Copy) {
   from 'resources'
   into 'target'
   include('**/*.txt', '**/*.xml', '**/*.properties')
}

3. 名称方法作为调用:

Groovy 中唯一支持的流行选项是简写符号

copy {
    from("resources")
    into("target")
    include("**/*.txt", "**/*.xml", "**/*.properties")
}
此选项会破坏任务配置避免,不建议使用!

无论选择哪种方法,任务都配置了要复制的文件名和文件的位置。

3. 实现任务

Gradle 提供了许多任务类型,包括 DeleteJavadocCopyExecTarPmd。如果 Gradle 没有提供满足您的构建逻辑需求的任务类型,您可以实现自定义任务类型。

要创建自定义任务类,您需要扩展 DefaultTask 并使扩展类抽象化

app/build.gradle.kts
abstract class MyCopyTask : DefaultTask() {

}
app/build.gradle
abstract class MyCopyTask extends DefaultTask {

}