# 任務(wù)
- 圖解 Gradle 構(gòu)建的三個(gè)階段
- 監(jiān)聽(tīng) Gradle 不同階段的回調(diào)瞎抛;
- 練習(xí)-計(jì)算 build 任務(wù)的執(zhí)行時(shí)長(zhǎng);
# Gradle 構(gòu)建過(guò)程
下面這張圖是是參考網(wǎng)上的却紧,感覺(jué)寫(xiě)的還不錯(cuò)桐臊。
根據(jù)在上圖中所示,Gradle 的構(gòu)建過(guò)程主要分為三個(gè)階段:
- 初始化階段
- 配置階段
- 執(zhí)行階段
## 監(jiān)聽(tīng) Gradle 初始化時(shí)機(jī)
在這個(gè)初始化階段中主要有兩個(gè)時(shí)機(jī)需要關(guān)注:
- setting.gradle 執(zhí)行結(jié)束的監(jiān)聽(tīng)
//1.setting.gradle 執(zhí)行結(jié)束的監(jiān)聽(tīng)
gradle.settingsEvaluated {
println "settings.gradle 初始化執(zhí)行結(jié)束"
}
- 參與構(gòu)建的Project對(duì)象創(chuàng)建完畢的監(jiān)聽(tīng)
//2.參與構(gòu)建的Project對(duì)象創(chuàng)建完畢的監(jiān)聽(tīng)
gradle.projectsLoaded {
Gradle gradle ->
println "settings.gradle 所有在 settings 中 include 的 Project 都創(chuàng)建完成了"
}
## 監(jiān)聽(tīng) Gradle 配置階段時(shí)機(jī)
在 settings.gradle 中可以配置需要參與構(gòu)建的 project 啄寡,并且在初始化階段就已經(jīng)創(chuàng)建好對(duì)應(yīng)的 project 實(shí)例了豪硅。
Gradle 的配置階段就是執(zhí)行每一個(gè) Project 對(duì)應(yīng)的 build.gradle 的配置代碼。
下面相關(guān)的API是關(guān)于配置的回調(diào):
- gradle.beforeProject
表示對(duì)
每一個(gè) project
在執(zhí)行配置代碼之前都會(huì)回調(diào)這個(gè)方法挺物。
gradle.beforeProject {
Project project ->
println ">>>>>>>>gradle beforeProject " + project.name + " 配置開(kāi)始前回調(diào)"
}
- project.beforeEvaluate
當(dāng)前 project
在執(zhí)行配置代碼之前的回調(diào)
通過(guò) gradle.afterProject 和 project.afterEvaluate 是差不多一下的懒浮。
this.afterEvaluate {
project ->
println project.name + " 配置結(jié)束監(jiān)聽(tīng)"
}
- gradle.projectsEvaluated
表示所有的 project 都執(zhí)行完對(duì)應(yīng)的 build.gradle 的配置代碼,準(zhǔn)備要去生成對(duì)應(yīng)的 Task 依賴圖识藤。
gradle.projectsEvaluated {
gradle ->
println "所有的project都配置完畢了砚著,準(zhǔn)備生成Task依賴關(guān)系"
}
- gradle.taskGraph.whenReady
表示 "task 依賴關(guān)系已經(jīng)生成"
gradle.taskGraph.whenReady {
TaskExecutionGraph graph ->
println "task 依賴關(guān)系已經(jīng)生成"
}
## 監(jiān)聽(tīng) Gradle 執(zhí)行階段時(shí)機(jī)
Gradle 在配置階段
中會(huì)執(zhí)行每一個(gè) project 的 build.gradle 的配置代碼,并且最終生成每一個(gè) Task 任務(wù)的依賴關(guān)系痴昧。下面到了執(zhí)行階段
就會(huì)根據(jù)這個(gè)依賴關(guān)系去執(zhí)行對(duì)應(yīng)的 Task 任務(wù)稽穆。
- gradle.taskGraph.beforeTask
每一個(gè) Task 任務(wù)執(zhí)行之前回調(diào)
gradle.taskGraph.beforeTask {
Task task ->
println "Project[${task.project.name}]--->Task[${task.name}] 在執(zhí)行之前被回調(diào)"
}
- gradle.taskGraph.afterTask
每一個(gè) task 執(zhí)行之后被回調(diào)
gradle.taskGraph.afterTask {
task, TaskState taskState ->
//第二個(gè)參數(shù)表示 task 的狀態(tài),是可選的參數(shù)
println "Project[${task.project.name}]--->Task[${task.name}] 在執(zhí)行完畢,taskState[upToDate:${taskState.upToDate},skipped:${taskState.skipped},executed:${taskState.executed},didWork:${taskState.didWork}]"
}
## Gradle 構(gòu)建執(zhí)行結(jié)束的回調(diào)
- gradle.buildFinished
當(dāng)所有的任務(wù)執(zhí)行完畢的回調(diào)
gradle.buildFinished {
BuildResult buildResult ->
println "構(gòu)建完畢"
}
練習(xí)-計(jì)算 build 任務(wù)的執(zhí)行時(shí)長(zhǎng)
在 Android 中 build 的執(zhí)行是最為復(fù)雜赶撰,也是執(zhí)行時(shí)間最久的一個(gè) Task 舌镶,因?yàn)?build 這個(gè)任務(wù)依賴了很多其他的任務(wù),第一個(gè)被依賴的任務(wù)是 preBuild 豪娜,因此我們希望在 preBuild 執(zhí)行之前記錄當(dāng)前的時(shí)間戳
taskStartTime
餐胀,在 build 執(zhí)行完畢之后記錄當(dāng)前的時(shí)間戳taskEndTime
,然后計(jì)算兩個(gè)時(shí)間戳的差值即是 build 任務(wù)的執(zhí)行時(shí)長(zhǎng)啦瘤载。
下面是具體的步驟:
- 通過(guò) project(":app") 對(duì) app module 單獨(dú)配置否灾。
- 通過(guò) project.afterEvaluate 監(jiān)聽(tīng) project 配置完畢。
- 通過(guò) getBuildTask 獲取 preBuild 和 build 兩個(gè)任務(wù)對(duì)象鸣奔。
- 監(jiān)聽(tīng) preBuild.doFirst 得到開(kāi)始執(zhí)行的時(shí)間戳墨技。
- 監(jiān)聽(tīng) build.doLast 得到執(zhí)行完畢的時(shí)間戳惩阶。
- 最后得到兩個(gè)時(shí)間戳的差值即是該任務(wù)的執(zhí)行市場(chǎng)。
- 執(zhí)行 build 任務(wù)扣汪。
//配置 app module
project(":app") {
Project project ->
project.afterEvaluate {
//獲取build task 任務(wù)
Task buildTask = getBuildTask(project, ":app:build")
//獲取 preBuild 任務(wù)
Task preBuildTask = getBuildTask(project, ":app:preBuild")
def taskStartTime = 0
def taskEndTime = 0
//在preBuild task 中追加一個(gè)監(jiān)聽(tīng)獲取在preBuild執(zhí)行之前的時(shí)間戳的 action
preBuildTask.doFirst {
taskStartTime = System.currentTimeMillis()
}
//在build task 中追加一個(gè)監(jiān)聽(tīng)獲取執(zhí)行完畢的時(shí)間戳的 action
buildTask.doLast {
taskEndTime = System.currentTimeMillis()
println "build task 執(zhí)行時(shí)間:${taskEndTime - taskStartTime}"
}
}
}
/**
* 獲取該project對(duì)應(yīng)的build任務(wù)
* @param project
* @return
*/
Task getBuildTask(Project project, String taskPath) {
//獲取該project管理task的容器
TaskContainer taskContainer = project.getTasks()
//拿到build的Task對(duì)象
Task buildTask = taskContainer.getByPath(taskPath)
return buildTask
}
執(zhí)行 build 任務(wù)
./gradlew build
記錄于 2018-07-26 晚