Gradle tip #1: tasks

时间:2022-09-09 14:59:14

With this post I would like to start series of Gradle-related topics I wish I knew when I first started writing Gradle build scripts.

以这篇博客开始,我将写一系列关于Gradle的文章,用来记录接触Gradle构建脚本以来我所理解的Gradle。

Today we will talk about Gradle tasks and specificallyconfiguration and execution parts of the task. Since these terms might appear unknown to vast majority of readers, it will be easier to have real examples. Essentially (sorry for looking ahead), we will try to figure out what is the difference between these 3 examples:

今天要讲的就是Gradle tasks以及task的配置和运行。可能有的读者还不了解Gradle task,用真实的例子来展示应该更容易被理解。下面的代码展示了三个Gradle task,稍后会讲解这三者的不同:

task myTask {
println "Hello, World!"
} task myTask {
doLast {
println "Hello, World!"
}
} task myTask << {
println "Hello, World!"
}

My goal - is to create a task which prints "Hello, World!" when I execute it.

我的目的是创建一个task,当它执行的时候会打印出来”Hello, World!”。

When I first started, my first guess was to implement it like this:

当我第一次创建task的时候,我猜测应该是这样来写的:

task myTask {
println "Hello, World!"
}

Now, let's try to execute my new task!

现在,试着来执行这个myTask,在命令行输入gradle myTask,打印如下:

user$ gradle myTask
Hello, World!
:myTask UP-TO-DATE

It seems to be working! It prints "Hello, World!".

这个task看起来起作用了。它打印了”Hello, World!”。

But! It doesn't work as we might expect it to work. And here is why. Let's try to call gradle tasks to see what other tasks are available:

但是,它其实并没有像我们期望的那样。下面我们来看看为什么。在命令行输入gradle tasks来查看所有可用的tasks。

user$ gradle tasks
Hello, World!
:tasks ------------------------------------------------------------
All tasks runnable from root project
------------------------------------------------------------ Build Setup tasks
-----------------
init - Initializes a new Gradle build. [incubating]
..........

Wait a second! Why my "Hello, World!" string is printed? I just called tasks, I didn't call my custom task!

等等,为什么”Hello, World!”打印出来了?我只是想看看有哪些可用的task,并没有执行任何自定义的task!

The reason why this is happening - is that Gradle task has 2 major stages in its lifecycle:

  • Configuration stage
  • Execution stage

原因其实很简单,Gradle task在它的生命周期中有两个主要的阶段:配置阶段 和 执行阶段。

I might not be super precise with terminology here, but this analogy helped me to understand tasks.

可能我的用词不是很精确,但这的确能帮助我理解tasks。

The thing is that Gradle has to configure all tasks specified in build script before actual build is started. It doesn't matter if certain task will be executed - it still needs to be configured.

Knowing that, how do I know which part of my task is evaluated during configuration and which one during execution?

And the answer is - the part specified within the top-level of the task - is task configuration section. I.e:

Gradle在执行task之前都要对task先进行配置。那么问题就来了,我怎么知道我的task中,哪些代码是在配置过程中执行的,哪些代码是在task执行的时候运行的?答案就是,在task的最顶层的代码就是配置代码,比如:

task myTask {
def name = "Pavel" //<-- this is evaluated during configuration
println "Hello, World!"////<-- this is also evaluated during configuration
}

That's why when I call gradle tasks I can see "Hello, World!" - this is our configuration section is executed. But that's not really what I want - I want "Hello, World!" to be printed only when I explicitly call my task.

这就是为什么我执行gradle tasks的时候,会打印出来”Hello, World!”-因为配置代码被执行了。但这并不是我想要的效果,我想要”Hello, World!”仅仅在我显式的调用myTask的时候才打印出来。

So how do I tell Gradle to do something when my tasks is executed?

In order to do that I need to specify task Action. The easiest way to specify task action is via Task#doLast() method:

为了达到这个效果,最简单的方法就是就是使用Task#doLast()方法。

task myTask {
def text = 'Hello, World!' //configure my task
doLast {
println text //this is executed when my task is called
}
}

Now my "Hello, World!" string will be printed only when I explicitly call gradle myTask

现在,”Hello, World!”仅仅会在我执行gradle myTask的时候打印出来。

Cool, now I know how to configure and make my task do real work only when I invoke it. What about that third option with <<symbol?:

Cool,现在我已经知道如何配置以及使task做正确的事情。还有一个问题,最开始的例子中,第三个task的<<符号是什么意思?

task myTask2 << {
println "Hello, World!"
}

This version is just a shortcut of doLast version, i.e. it is exactly the same as I would write:

这其实只是doLast的一个语法糖版本。它和下面的写法效果是一样的:

task myTask {
doLast {
println 'Hello, World!' //this is executed when my task is called
}
}

However, since now everything goes into execution part, I cannot configure my task the same way I did it with doLastoption (it is still possible to do, but in a slightly different way). So this option is good for really small tasks which do not require configuration, but if you have some task other than printing a "Hello, World!" - you might consider going with doLast.

但是,这种写法所有的代码都在执行部分,没有配置部分的代码,因此比较适合那些简小不需要配置的task。一旦你的task需要配置,那么还是要使用doLast的版本。

Happy gradling!

我是天王盖地虎的分割线

http://trickyandroid.com/gradle-tip-1-tasks/

http://blog.csdn.net/lzyzsd/article/details/46934187