Gradle入門系列(四)——初識Gradle Task

Gradle中的Task

1、Task定義及配置

TaskContainer:管理全部的Task,如:增長、查找。html

定義(建立)Taskapi

// 直接經過task函數去建立
task helloTask {
  println 'i am helloTask.'
}

// 經過TaskContainer去建立
this.tasks.create(name: 'helloTask2') {
  println 'i am helloTask 2.'
}
複製代碼

配置Task數組

// 給Task指定分組與描述
task helloTask(group: 'study', description: 'task study'){ // 語法糖
  ...
}
task helloTask {
  group 'study' // setGroup('study')
  description 'task study' // setDescription('task study')
  ...
}
複製代碼

Task除了能夠配置group、description外,還能夠配置name、type、dependsOn、overwrite、action。閉包

結論:app

  • 給Task分組以後,該task會被放到指定組中,方便歸類查找。(默認被分組到other中)
  • 給Task添加描述,至關於給方法添加註釋。

2、Task的執行詳情

Task中doFirst與doLast的使用:函數

task helloTask {
  println 'i am helloTask.'
  doFirst {
    println 'the task group is: ' + group
  }
  // doFirst、doLast能夠定義多個
  doFirst {}
}
// 外部指定doFirst(會比在閉包內部指定的doFirst先執行)
helloTask.doFirst {
  println 'the task description is: ' + description
}

// 統計build執行時長
def startBuildTime, endBuildTime
this.afterEvaluate { Project project ->
  // 保證要找的task已經配置完畢
  def preBuildTask = project.tasks.getByName('preBuild') // 執行build任務時,第一個被執行的Task
  preBuildTask.doFirst {
    startBuildTime = System.currentTimeMillis()
  }
  def buildTask = project.tasks.getByName('build') // 執行build任務時,最後一個被執行的Task
  buildTask.doLast {
    endBuildTime = System.currentTimeMillis()
    println "the build time is: ${endBuildTime - startBuildTime}"
  }
}
複製代碼

結論:gradle

  • Task閉包中直接編寫的代碼,會在配置階段執行。能夠經過doFirst、doLast塊將代碼邏輯放到執行階段中執行。
  • doFirst、doLast能夠指定多個。
  • 外部指定的doFirst、doLast會比內部指定的先執行。
  • doFirst、doLast能夠對gradle中提供的已有的task進行擴展。

3、Task的執行順序

task執行順序指定的三種方式:ui

  1. dependsOn強依賴方式
  2. 經過Task輸入輸出指定(與第1種等效)
  3. 經過API指定執行順序

一、Task的依賴

// ============= dependsOn強依賴方式 =============
task taskX {
  doLast {
      println 'taskX'
  }
}
task taskY {
  doLast {
      println 'taskY'
  }
}
// 方式一:靜態依賴
// task taskZ(dependsOn: taskY) // 依賴一個task
task taskZ(dependsOn: [taskX, taskY]) { // 依賴多個task,須要用數組[]表示
  doLast {
      println 'taskZ'
  }
}
// 方式二:靜態依賴
taskZ.dependsOn(taskX, taskY)
// 方式三:動態依賴
task taskZ() {
  dependsOn this.tasks.findAll {    // 依賴全部以lib開頭的task
    task -> return task.name.startsWith('lib')
  }
  doLast {
      println 'taskZ'
  }
}
複製代碼

其餘:this

  • taskZ依賴了taskX與taskY,因此在執行taskZ時,會先執行taskX、taskY。
  • taskZ依賴了taskX與taskY,但taskX與taskY沒有關係,它們的執行順序是隨機的。

二、Task的輸入輸出

inputs和outputs是Task的屬性。 inputs能夠是任意數據類型對象,而outputs只能是文件(或文件夾)。 TaskA的outputs能夠做爲TaskB的inputs。lua

例子:writeTask輸入擴展屬性,輸出文件,readTask輸入writeTask的輸出文件

ext {
    versionCode = '1.0.0'
    versionName = '100'
    versionInfo = 'App的第1個版本,完成聊天功能'
    destFile = file('release.xml')
    if (destFile != null && !destFile.exists()) {
        destFile.createNewFile()
    }
}

task writeTask {
    inputs.property('versionCode', this.versionCode)
    inputs.property('versionName', this.versionName)
    inputs.property('versionInfo', this.versionInfo)
    outputs.file this.destFile
    doLast {
        def data = inputs.getProperties() // 返回一個map
        File file = outputs.getFiles().getSingleFile()
        // 將map轉爲實體對象
        def versionMsg = new VersionMsg(data)
        def sw = new StringWriter()
        def xmlBuilder = new MarkupBuilder(sw)
        if (file.text != null && file.text.size() <= 0) { // 文件中沒有內容
            // 實際上,xmlBuilder將xml數據寫入到sw中
            xmlBuilder.releases { // <releases>
                release { // <releases>的子節點<release>
                    versionCode(versionMsg.versionCode)
                    // <release>的子節點<versionCode>1.0.0<versionCode>
                    versionName(versionMsg.versionName)
                    versionInfo(versionMsg.versionInfo)
                }
            }
            // 將sw裏的內容寫到文件中
            file.withWriter { writer ->
                writer.append(sw.toString())
            }
        } else { // 已經有其它版本信息了
            xmlBuilder.release {
                versionCode(versionMsg.versionCode)
                versionName(versionMsg.versionName)
                versionInfo(versionMsg.versionInfo)
            }
            def lines = file.readLines()
            def lengths = lines.size() - 1
            file.withWriter { writer ->
                lines.eachWithIndex { String line, int index ->
                    if (index != lengths) {
                        writer.append(line + '\r\n')
                    } else if (index == lengths) {
                        writer.append(sw.toString() + '\r\n')
                        writer.append(line + '\r\n')
                    }
                }
            }
        }
    }
}

task readTask {
    inputs.file destFile
    doLast {
        def file = inputs.files.singleFile
        println file.text
    }
}

task taskTest(dependsOn: [writeTask, readTask]) {
    doLast {
        println '任務執行完畢'
    }
}

class VersionMsg {
    String versionCode
    String versionName
    String versionInfo
}
複製代碼

經過執行 gradle taskTask 以後,就能夠在工程目錄下看到release.xml文件了。

結論:

  • 由於writeTask與readTask經過inputs、outputs產生了關聯關係,因此,readTask必定會在writeTask執行以後才執行。

三、Task API指定順序

task指定執行順序的api有:

  • mustRunAfter : 強行指定在某個或某些task執行以後才執行。
  • shouldRunAfter : 與mustRunAfter同樣,但不強制。
task taskX {
    doLast {
        println 'taskX'
    }
}
task taskY {
    // shouldRunAfter taskX
    mustRunAfter taskX
    doLast {
        println 'taskY'
    }
}
task taskZ {
    mustRunAfter taskY
    doLast {
        println 'taskZ'
    }
}
複製代碼

經過執行 gradle taskY taskZ taskX 以後,能夠看到終端仍是按taskX、taskY、taskZ順序執行的。

4、掛接到構建生命週期

例子:build任務執行完成後,執行一個自定義task

this.afterEvaluate { Project project ->
    def buildTask = project.tasks.getByName('build')
    if (buildTask == null) throw GradleException('the build task is not found')
    buildTask.doLast {
        taskZ.execute()
    }
}
複製代碼

例子:Tinker將自定義的manifestTask插入到了gradle腳本中processManifest與processResources這兩個任務之間

TinkerManifestTask manifestTask = project.tasks.create("tinkerProcess${variantName}Manifest", TinkerManifestTask)
...
manifestTask.mustRunAfter variantOutput.processManifest
variantOutput.processResources.dependsOn manifestTask
複製代碼

5、Task類型

Gradle DSL Version 5.1

Copy - Gradle DSL Version 5.1

相關文章
相關標籤/搜索