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;
- 任务执行结束。