Project和Task是Gradle中最核心的两个元素,当创建了一个build.gradle脚本那么Gradle便会依据脚本的配置创建一个Project,而脚本中的Task也会创建与之相对应DeafultTask实现。
Gradle通过一个个Task来完成具体的构建任务 ,可以说Task乃是Gradle的核心。根据执行阶段的不同,可以将Gradle中Task分为配置型Task以及动作型Task。
一、Task的基本组成
查看Task DSL说明我们可以发现,通常一个Task包括: 依赖、动作、属性、输入、输出、终结器等构成。
当然这些并不是每一个都是必须的。创建任务的时候需要根据自己的具体需要进行选择性配置。
二、定义Task
Gralde中定义一个任务的方式十分灵活,下面都是定义一个任务的方式。
task myTask
task myTask { configure closure }
task myTask(type: SomeType)
task myTask(type: SomeType) { configure closure }
配置型Task
定义一个Task十分简单,只需要使用task name{}即可定义一个简单的任务。
task helloTask {
println "Hello World"
}
执行gradle hT,可以观察到执行结果。这里你可能会注意到Hello world并非在执行阶段开始执行的,而是在配置阶段就已经打印了。没错,这就是一个配置型Task,因为Gradle在任务执行前,总会去遍历所有任务去生成一张DAG(有向无环图)来确定任务之间的关系。
动作型Task
如果不想让任务在配置阶段执行,那么可以参照如下方式,通过给任务添加action的方式使其在执行阶段运行。
task helloTask {
doLast {
println "Hello World"
}
}
常用的action有两个,doFirst和doLast,通过这两个见名知意的action可以用来定置化你的任务行为。
- 一个Task包含若干
Action。所以,Task有doFirst和doLast两个函数,用于添加需要最先执行的Action和需要和需要最后执行的Action。Action就是一个闭包。 - Task创建的时候可以指定Type,通过type:名字表达。这是什么意思呢?其实就是告诉
Gradle,这个新建的Task对象会从哪个基类Task派生。比如,Gradle本身提供了一些通用的Task,最常见的有Copy 任务。Copy是Gradle中的一个类。当我们:*task myTask(type:Copy)*的时候,创建的Task就是一个Copy Task。 - 当我们使用
task myTask{ xxx}的时候。花括号是一个closure。这会导致gradle在创建这个Task之后,返回给用户之前,会先执行closure的内容。
当用户执行test任务时,执行以下步骤:
- 执行build.gradle,初始化任务,初始化步骤为2-7;
- 注册test任务,任务体都是默认好的,不可更改,为打印冒号加任务名,所以test的任务体为
println ':test'; - 每个任务都有一个队列一个栈,一个是依赖队列,一个是first栈,一个是last队列;
- 遇到doFirst函数,该函数接受一个闭包作为参数,doFirst函数会把闭包参数放入first栈;
- 遇到doLast函数,同理doFirst函数,闭包参数会被放入last队列;
- 然后遇到
println 'hello.',执行该指令//1; - 至此初始化过程完成。
- 开始执行test任务前,先检查是否任务依赖,如果有,先把依赖的任务都执行完。
- 然后开始执行test任务;
- 执行时,首先执行test任务的任务体,即
println ':test'; - 然后执行test任务的first栈//2和last队列//3;
- 任务执行结束。