學(xué)習(xí)本系列前可以下載相關(guān)的github項(xiàng)目gradleLearnDemo斤彼。
地址:https://github.com/sososeen09/gradleLearnDemo
Gradle的增量式構(gòu)建特性緊緊的與生命周期相結(jié)合糕再。Gradle的增量式構(gòu)建支持自動(dòng)鑒別不需要被運(yùn)行的任務(wù)柱查。這些任務(wù)會(huì)被標(biāo)記為 UP-TO-DATE瞭郑。特別是在大型的企業(yè)級(jí)項(xiàng)目漾根,這個(gè)特性是節(jié)約時(shí)間的好幫手冬殃。
Gradle通過比較兩次構(gòu)建task的inputs和outputs來決定task是否是最新的霎烙。自從最后一次task執(zhí)行以來,如果inputs和outputs沒有發(fā)生變化赡茸,則認(rèn)為task是最新的缎脾。
輸入可以是一個(gè)目錄、一個(gè)或多個(gè)文件占卧,或者是一個(gè)任意屬性遗菠。
一個(gè)task的輸出是通過一個(gè)目錄或1~n個(gè)文件來定義的。
inputs和outputs在DefaultTask類中被定義為屬性或者有一個(gè)直接類來表示华蜒。對(duì)應(yīng)的是TaskInputs和TaskOutputs辙纬。
為了便于表述,我們還是接著上一篇的那個(gè)例子叭喜,從配置文件version.properties中讀取參數(shù)贺拣,并打印出版本號(hào):
version=new ProjectVersion(0,1)
class ProjectVersion{
Integer major
Integer minor
Boolean release
ProjectVersion(Integer major, Integer minor){
this.major=major
this.minor=minor
this.release=Boolean.FALSE
}
ProjectVersion(Integer major, Integer minor, Boolean release){
this.major=major
this.minor=minor
this.release=release
}
@Override
String toString(){
"$major.$minor${release?'': '-SNAPSHOT'}"
}
}
task printVersion << {
logger.quiet "Version: $version"
}
// Project接口提供了file方法,它會(huì)創(chuàng)建一個(gè)相對(duì)于項(xiàng)目目錄的java.io.File實(shí)例
ext.versionFile=file('version.properties')
task loadVersion{
project.version=readVersion()
}
//readVersion方法捂蕴,與task是不同的
ProjectVersion readVersion(){
logger.quiet 'Reading the version file'
if(!versionFile.exists()){
throw new GradleException ("Required version file dose not exist:$versionFile.canonicalPath " )
}
//Groovy的文件實(shí)現(xiàn)通過添加新的方法來讀取InputStream
Properties versionProps=new Properties()
versionFile.withInputStream{stream->
versionProps.load(stream)
}
// 在Groovy中譬涡,如果return是方法中的最后一條語句的話,則可以將它省略
new ProjectVersion(versionProps.major.toInteger(),versionProps.minor.toInteger(),versionProps.release.toBoolean())
}
配置文件中的初始參數(shù)是這樣的:
major=0
minor=1
release=false
現(xiàn)在啥辨,我們要增加一個(gè)新的任務(wù):新建一個(gè)名為makeReleaseVersion的task來將配置文件中的release屬性改為true涡匀。
task makeReleaseVersion(group :'versioning',description :'Makes project a release version.' )<< {
version.release=true
// Ant task 的propertyfile 提供了一種便利的方式來修改屬性文件
ant.propertyfile(file:versionFile){
entry(key:'release',type:'string',operation:'=',value: 'true')
}
}
執(zhí)行g(shù)radle makeReleaseVersion
之后打開配置文件 version.properties 就可以看到文件中的release屬性已經(jīng)變?yōu)閠rue了。
執(zhí)行 gradle printVersion
命令溉知,可以看到結(jié)果
:printVersion
Version: 0.1
注意陨瘩,雖然我們將配置文件中的release標(biāo)記為true腊嗡,但是Gradle并不知道。當(dāng)我們?cè)俅螆?zhí)行makeReleaseVersion這個(gè)task的時(shí)候拾酝,doLast這個(gè)閉包中的代碼還是會(huì)執(zhí)行,會(huì)花多余的時(shí)間卡者。為了解決這個(gè)問題蒿囤,需要聲明它的inputs和outputs。
task makeReleaseVersion(group :'versioning',description :'Makes project a release version.' ){
//在配置階段聲明inputs/outputs
//聲明版本的release屬性作為輸入
inputs.property('release',version.release)
//由于版本文件被修改了崇决,所以它被聲明作為輸出文件屬性
outputs.file versionFile
doLast{
version.release=true
// Ant task 的propertyfile 提供了一種便利的方式來修改屬性文件
ant.propertyfile(file:versionFile){
entry(key:'release',type:'string',operation:'=',value: 'true')
}
}
}
記撞姆獭:task的inputs和outputs屬性是在配置階段執(zhí)行的用來連接task的依賴。這就是他們需要在配置塊中被定義的原因恒傻。為了避免出現(xiàn)所不期望出現(xiàn)的行為脸侥,請(qǐng)確保賦給inputs和outputs的值在配置階段是可訪問的。如果需要通過編程獲得輸出盈厘,可以通過TaskOutputs上的upToDateWhen(Closure)方法來實(shí)現(xiàn)睁枕。與常規(guī)的inputs和outputs相比,這個(gè)方法是在執(zhí)行期間執(zhí)行的沸手。如果閉包返回true外遇,這個(gè)task則會(huì)被認(rèn)為是最新的。
如果執(zhí)行兩次makeReleaseVersion任務(wù)契吉,會(huì)發(fā)現(xiàn)Gradle已經(jīng)知道配置文件中的release 已經(jīng)變?yōu)閠rue跳仿,會(huì)自動(dòng)跳過第二次執(zhí)行。
:makeReleaseVersion UP-TO-DATE
實(shí)際上捐晶,我們?cè)赼pk打包過程中經(jīng)撤朴铮看到的UP-TO-DATE 就是因?yàn)檩斎胼敵鰶]有發(fā)生變化,Gradle跳過了這個(gè)task的執(zhí)行惑灵。
下一篇山上,我們開始學(xué)習(xí)Gradle中如何掛接到構(gòu)建生命周期。