GRADLE構建最佳實踐
隨著谷歌對Eclipse的無情拋棄和對Android Studio的日趨完善,使用gradle構建Android項目已經成為開發者的一項必會良技。那么,問題來了,采用什么樣的姿勢才能讓項目開發構建過程高潮迭起,精彩不斷呢?
其實網上有很多關于gradle的文章,gradle官方和谷歌也提供了詳細的文檔和教程,可素,當你在構建過程中遇到一些問題或者有特殊的愛好(需求)的時候,這些東西未必能幫(mei)上(shen)什(me)么(niao)忙(yong),然后就是一頓KX上網找谷歌蜀黍約約約,去stackoverflow上各種搜刮問大神,最后解決了。即使沒有真的解決那么就忍了。
那怎么行?是可忍孰不可忍,奇技淫巧必須有。所以就會有這樣一篇文章,我在這里不講原理,因為我知道很多人明白辣么多的底層原理,仍然擼不上好代碼,做不成好項目,出不了好產品,自然也就過不好這一生咯。
我們先從GRADLE構建的時間花銷開始談起。
加速篇
GRADLE的構建過程通常會比較漫長,一個中等項目,10M左右大小的app,一次完整構建大概在5分鐘左右,是不是很嚇人,當然,如果是在調試階段,采用Android Studuo 2.0,默認提供的Instant Run方式,每次修改都不會重新構建項目,從而加快了構建過程。恩,這是另一個故事,這里,我們先談談GRADLE腳本的加速姿勢。
一般來說,GRADLE一次完整的構建過程通常分成三個部分,初始化,配置和執行任務,那么我們可以考慮從這三個部分分別嘗試優化。
使用daemon
構建初始化的很多工作是關于java虛擬機的啟動,加載虛擬機環境,加載class文件等,如果這些動作交給一個單獨的后臺進程去做,那么,第一次初始化之后的修改代碼再構建是不是可以節省很多時間呢?答案是肯定的,通過在gradle.properties加入這樣一句來開啟,如果想讓修改全局所有項目都生效,那么修改這個文件~/.gradle/gradle.properties
org.gradle.daemon=true
按需配置
配置有一種方式是按需配置,目前還在試驗孵化階段,所以默認是關閉的,可以通過在gradle.properties加入這樣一句來開啟
org.gradle.configureondemand=true
依賴庫使用固定版本
項目開發過程中,難免需要用到三方庫,這就形成了項目之間的依賴關系,GRADLE提供了多種集成三方庫的方式,提供了很方便的項目依賴管理,本地庫,庫工程,maven庫全支持。既然用到庫,就會遇到庫版本的問題和升級問題,其中maven庫的依賴管理支持一種動態版本的方式,也就是說,GRADLE可以做到不依賴具體某個版本的庫,而是每次從repo拉取最新的庫到本地做編譯。具體使用是這樣的:
拿gson庫舉例,如果依賴2.2.1這個版本,可以在build.gradle文件里這樣寫
dependencies {
compile 'com.google.code.gson:gson:2.2.1'
}
如果不想依賴具體的庫,想每次從maven repo中拉取最新的庫,那么,可以寫成這樣:
dependencies {
compile 'com.google.code.gson:gson:2.2.+'
}
也可以寫成這樣
dependencies {
compile 'com.google.code.gson:gson:2.+'
}
甚至可以這樣
dependencies {
compile 'com.google.code.gson:gson:+'
}
其中含義相信不用我解釋,大家也看得明白吧。
用”+”來通配一個版本族,這樣有個好處是maven上有新庫了,不用你操心升級,GRADLE編譯的時候自動升級了,但是帶來了兩個壞處,一是,有可能新版庫的接口改了,導致編譯失敗,這個時候需要修改代碼做升級適配;更大的壞處是,每次GRADLE編譯完整的項目,都會去maven上試圖拉取最新的庫,這樣,拖慢了編譯速度,尤其在網絡非常差的時候,所以,為了構建速度,建議寫死依賴庫的版本號。
升級到最新的GRADLE和JDK
有一個很通俗的道理是,發展的東西會越來越好,最新版的GRADLE和JDK往往是性能最好,運行最流暢最快的,所以,升級吧,JDK的升級這里不說了,具體看Oracle的官方文檔。這里說說GRADLE的版本升級,GRALDE采用了一種叫做wrapper的方式,可以做到每個項目獨立使用其自己的GRADLE版本,這樣做的好處不言而喻,每個項目的構建環境獨立,互不影響。但為什么會出現這個東西,我的猜想是因為GRADLE發展太快,新舊版本之間很難兼容。如果你有多個項目都采用GRADLE構建,假設都用同一個全局的GRADLE,那么當這個GRADLE升級后,所有的項目可能都會編譯失敗,你得一個一個改配置,那么,下次再升級,同樣的流程的再走一遍,是不是很煩。采用wrapper的方式很好的解決了這個問題,每個項目采用獨立的GRADLE版本,互不影響,如果你只想升級其中一個,你改這一個項目的GRADLE wrapper就好了。在你的項目目錄下找到這個文件gradle/wrapper/gradle-wrapper.properties并修改distributionUrl=https://services.gradle.org/distributions/gradle-2.11-all.zip到你想升級的版本就可以了。
減少編譯腳本中的I/O操作
有時候,編譯腳本中會有一些代碼做動態信息的獲取,比如想從git中獲取一個數字作為版本號
def cmd = 'git rev-list HEAD --first-parent --count'
def gitVersion = cmd.execute().text.trim().toInteger()
android {
defaultConfig {
versionCode gitVersion
}
}
其實這個操作主要是為了在構建的機器上為了發布版本而做的,日常環境研發調試無需這樣,所以可以優化成如下方式:
def gitVersion() {
if (!System.getenv('CI_BUILD')) {
// don't care
return 1
}
def cmd = 'git rev-list HEAD --first-parent --count'
cmd.execute().text.trim().toInteger()
}
android {
defaultConfig {
versionCode gitVersion()
}
}
并行構建模塊化項目
將你的項目拆分成多個子項目并開啟并行構建也是一個不錯的主意,比如將相對獨立的模塊拆分成獨立的庫工程(Library projects),主工程(Application project)依賴這些庫工程,這樣的話,開啟并行構建才會發揮作用。并行構建開啟方式是修改文件gradle.properties,加入如下行:
org.gradle.parallel=true
基礎配置篇
全局基礎配置管理
#### 設置全局編碼 如果導入一個windows下編寫的項目,而代碼中有中文注釋,采用GBK, GB18030等編碼方式時,編譯會報錯,可以采用如下方式統一項目的編碼
allprojects {
repositories {
jcenter()
}
tasks.withType(JavaCompile) {
options.encoding = "UTF-8"
}
}
設置全局編譯器的版本
如果編程過程中采用了新版JDK(比如1.7)才支持的特性(比如new HashMap<>這樣的寫法),而編譯的時候默認是舊版的JDK(比如1.6),這個時候編譯會報錯,采用如下方式可以指定用哪個版本的編譯器編譯,前提是JAVA_HOME指定的JDK是大于等于新版JDK的哦^o^,其他和java編譯器相關的也可以在這里配置
allprojects {
repositories {
jcenter()
}
tasks.withType(JavaCompile) {
sourceCompatibility = JavaVersion.VERSION_1_7
targetCompatibility = JavaVersion.VERSION_1_7
}
}
如果不想全局生效,可以將tasks.withType(JavaCompile)放入某個子項目中。
配置簽名信息
簽名信息屬于敏感信息,建議不要寫死放到gradle腳本中,而是寫到一個單獨的配置文件里,而且這個配置文件不要同步到版本管理系統上,而是由本地維護,防止在版本管理平臺上泄漏敏感信息。建議簽名信息內容寫到gradle.properties或者local.properties文件里,這樣,gradle腳本可以直接引用,如果是放在一個自定義的文件中,gradle腳本需要提供相應的代碼來讀取文件的內容。 文件內容參考如下:
RELEASE_KEY_PASSWORD=android
RELEASE_KEY_ALIAS=androidreleasekey
RELEASE_STORE_PASSWORD=android
RELEASE_STORE_FILE=../resources/release.keystore
DEBUG_KEY_PASSWORD=android
DEBUG_KEY_ALIAS=androiddebugkey
DEBUG_STORE_PASSWORD=android
DEBUG_STORE_FILE=../resources/debug.keystore
gradle腳本引用代碼參考:
android {
signingConfigs {
debug {
storeFile file(DEBUG_STORE_FILE)
storePassword DEBUG_STORE_PASSWORD
keyAlias DEBUG_KEY_ALIAS
keyPassword DEBUG_KEY_PASSWORD
}
release {
storeFile file(RELEASE_STORE_FILE)
storePassword RELEASE_STORE_PASSWORD
keyAlias RELEASE_KEY_ALIAS
keyPassword RELEASE_KEY_PASSWORD
}
}
}
如果簽名信息沒有放到gradle.properties或者local.properties文件里,那就需要自己寫代碼讀取咯,假設簽名信息放在signing.properties文件中, 文件內容可以參考上面,讀取文件的代碼放入gradle腳本中就可以了,參考代碼如下
def File propFile = new File('signing.properties')
if (propFile.canRead()) {
def Properties props = new Properties()
props.load(new FileInputStream(propFile))
if (props != null && props.containsKey('RELEASE_STORE_FILE') && props.containsKey('RELEASE_STORE_PASSWORD') &&
props.containsKey('RELEASE_KEY_ALIAS') && props.containsKey('RELEASE_KEY_PASSWORD')) {
android.signingConfigs.release.storeFile = file(props['RELEASE_STORE_FILE'])
android.signingConfigs.release.storePassword = props['RELEASE_STORE_PASSWORD']
android.signingConfigs.release.keyAlias = props['RELEASE_KEY_ALIAS']
android.signingConfigs.release.keyPassword = props['RELEASE_KEY_PASSWORD']
println 'all good to go'
} else {
android.buildTypes.release.signingConfig = null
println 'signing.properties found but some entries are missing'
}
} else {
println 'signing.properties not found'
android.buildTypes.release.signingConfig = null
}
設置第三方maven地址
其中name和credentials是可選項,視具體情況而定
allprojects {
repositories {
maven {
url 'url'
name 'maven name'
credentials {
username = 'username'
password = 'password'
}
}
}
}
GRADLE腳本拆分以及引用
如果一個gradle腳本太大,可以按照具體任務的類型拆分成幾個子腳本,然后引入到主腳本中
apply from:"../resource/config.gradle"
全局變量定義及引用
可以在頂層build.gradle腳本中定義一些全局變量,提供給子腳本引用
ext {
// global variables definition
compileSdkVersion = 'Google Inc.:Google APIs:23'
buildToolsVersion = "23.0.2"
minSdkVersion = 14
targetSdkVersion = 23
}
子腳本引用
android {
compileSdkVersion rootProject.ext.compileSdkVersion
buildToolsVersion rootProject.ext.buildToolsVersion
defaultConfig {
minSdkVersion rootProject.ext.minSdkVersion
targetSdkVersion rootProject.ext.targetSdkVersion
}
}
構建參數篇
構建參數設置
AndroidManifest占位符,BuildConfig以及資源配置
根據版本類型和構建變種定義不同的變量值供程序引用
manifestPlaceholders = [APP_KEY:"release"]
buildConfigField "String", "EMAIL", "\"release@android.studio.com\""
resValue "string", "content_main", "Hello world from release!"
buildConfigField支持Java中基本數據類型,如果是字符串,記得加轉義后的雙引號 resValue支持res/values下的資源定義,字符串無需叫轉義后的雙引號
設置支持的語言
利用這個配置可以去掉三方庫中無用的語言
android {
defaultConfig {
resConfigs "zh"
}
}
重命名產出的文件
需要將產出的aar/apk移到另外一個地方的時候
android.libraryVariants.all { variant ->
variant.outputs.each { output ->
if (output.outputFile != null && output.outputFile.name.endsWith('.aar')) {
def name = "${rootDir}/demo/libs/library.aar"
println name
output.outputFile = file(name)
}
}
}
刪除unaligned apk
刪除無用的apk中間產物
android.libraryVariants.all { variant ->
variant.outputs.each { output ->
if (output.zipAlign != null) {
output.zipAlign.doLast {
output.zipAlign.inputFile.delete()
}
}
}
}
將自定義的任務加入構建流程
有時候編寫了一些自定義的任務,希望加入到構建流程中對輸入做預處理或者對輸出做后處理
project.tasks.whenTaskAdded { task ->
android.applicationVariants.all { variant ->
if (task.name == "prepare${variant.name.capitalize()}Dependencies") {
task.dependsOn ":library:assemble${variant.name.capitalize()}"
}
}
}
比如這里app工程依賴library的構建,可以這樣手工指定依賴關系
打包選項
有時候引用的三方庫會帶有一些配置文件xxxx.properties,或者license信息,打包的時候想去掉這些信息,就可以這樣做
android {
packagingOptions {
exclude 'proguard-project.txt'
exclude 'project.properties'
exclude 'META-INF/LICENSE.txt'
exclude 'META-INF/LICENSE'
exclude 'META-INF/NOTICE.txt'
exclude 'META-INF/NOTICE'
exclude 'META-INF/DEPENDENCIES.txt'
exclude 'META-INF/DEPENDENCIES'
}
}
lint選項開關
lint會按默認選項會做嚴格檢查,遇到包錯誤會終止構建過程,可以用如下開關關掉這個選項,不過最好是重視下lint的輸出,有問題及時修復,避免潛在問題
android {
lintOptions {
abortOnError false
}
}
依賴庫篇
三方庫(本地,maven)的依賴和工程庫依賴關系
依賴庫按構建目標定制
不同的依賴庫可以按構建目標做定制,比如freemium的變種集成了廣告,就可以這樣寫
dependencies {
freemiumCompile 'com.google.android.gms:ads:7.8.0'
}
aar本地庫依賴
jar本地庫的依賴很容易寫,arr本地庫的依賴稍微麻煩些
allprojects {
repositories {
jcenter()
flatDir {
dirs 'libs'
}
}
}
dependencies {
compile(name:'本地庫aar的名字,不用加后綴', ext:'aar')
}
NDK篇
NDK配置
只保留某一個abi,比如arm-eabi
為了包大小的考慮,去掉多余的本地庫
android {
defaultConfig {
ndk {
abiFilters 'armeabi'
}
}
}
總結篇
只有更好,木有最好;
只有總結,木有完結;
筆者從軟件時代開始使用構建工具和系統,一路從mk, make, cmake, qmake, 再到ant,maven, ivy, 到如今互聯網時代的gradle,sbt,構建配置越來越抽水馬桶化的人性體驗和更多的功能,讓開發者更專注于自己的業務開發,每個人都在自己的崗位專注的精耕細作專業的事,這樣才會有更高效的產出和成果,用好手頭的每一個工具,掌握各種姿勢和適用場景,這會是高效率高質量開發的良好的開端。
希望大家的姿勢都用對了,妥妥的,新姿勢,新場景,也請反饋給我,謝謝!
感謝篇
加速GRADLE構建的6個技巧
安卓新的構建系統
GRADLE官網
來源:http://www.figotan.org/2016/04/01/gradle-on-android-best-practise/