Gradle小知识#1:tasks

从这篇博文开始我打算开启关于Gradle相关知识的一些列博文。现在想想,如果我刚开始接触Gradle的时候知道这些知识的话那该多好啊。

今天我们来聊聊Gradle的任务,尤其是任务的配置和执行部分。因为这块知识对很多读者来说还不是很清楚,所以通过一个真实的例子来说明下就再好不过了。大体上(抱歉有点超前)我们是想要弄清楚下面这三个例子到底有啥区别:

1
2
3
4
5
6
7
8
9
10
11
12
13
task myTask {
println "Hello, World!"
}

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

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

我的目的是创建一个任务,当这个任务被执行的时候它能打印”Hello, World!”。一开始,我是这样实现的:

1
2
3
task myTask {
println "Hello, World!"
}

现在我们执行下这个任务!

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

这好像没啥问题。正确的打印出了“Hello, World!”
但是,这并没有按照我们预想的那样工作。下面让我告诉你为什么。我们试着调用下gradle tasks看看还有其他什么能执行的任务没:

1
2
3
4
5
6
7
8
9
10
11
12
user$ gradle tasks
Hello, World!
:tasks

------------------------------------------------------------
All tasks runnable from root project
------------------------------------------------------------

Build Setup tasks
-----------------
init - Initializes a new Gradle build. [incubating]
..........

等等!为什么打印了”Hello, World!”?我只是调用了tasks,我并没有调用我定义的任务呀!

原因就是在Gradle任务的生命周期中,有两个重要的阶段:

  • 配置阶段
  • 执行阶段

我对这些专业术语可能不太熟悉,但类比一下可以帮助我更好的理解Gradle的任务。

事情是这样的,Gradle在真正进行构建之前必须把构建脚本中定义的所有任务配置一下。不管这个任务会不会被执行,它都要先配置一下。

既然了解了这一点,但我又该怎么知道任务中哪部分是在配置阶段执行,哪部分又是在执行阶段执行呢?答案就是:任务中的顶层部分是配置阶段会执行的代码。比如:

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

这就是为什么我只是调用了gradle tasks,但却打印了”Hello, World!”。因为这部分代码在配置阶段执行了。但这并不是我想要的。我想要的是这段代码只有在我调用我的任务的时候才去执行。

那么,我该怎么告诉Gradle当我执行我的任务的时候才去做一些事情呢?

这时我就需要给任务指定一个动作。最简单的做法就是通过Task#doLast())方法:

1
2
3
4
5
6
task myTask {
def text = 'Hello, World!' //configure my task
doLast {
println text //this is executed when my task is called
}
}

现在,”Hello, World!”只有在我显式的调用gradle myTask的时候才会打印出来。

Cool!现在我知道怎么配置任务并且让任务只有在我明确调用的时候才会做真正的工作。那么第三个例子中的<<又是啥?:

1
2
3
task myTask2 << {
println "Hello, World!"
}

这只是doLast的简写版本而已。这跟下面的写法完全一样:

1
2
3
4
5
task myTask {
doLast {
println 'Hello, World!' //this is executed when my task is called
}
}

但是,既然所有的代码都在执行部分了,那我就不能像doLast写法那样配置我的任务了(放心吧,还是可以的,只是稍微有点不同而已)。这种简写方式对那些不需要进行配置的小任务来说很方便。但是,如果你的任务不仅仅像打印”Hello, World!”这么简单的话,那还是建议考虑使用doLast这种写法。