在平時(shí)的Android開發(fā)中,我們經(jīng)常會(huì)遇到在不同網(wǎng)絡(luò)環(huán)境(比如:開發(fā)環(huán)境、測(cè)試環(huán)境)之間的切換、一次打多個(gè)渠道包等需求,如何優(yōu)雅的管理網(wǎng)絡(luò)環(huán)境的配置?如何快速的打出多個(gè)渠道包?這是一個(gè)值得研究的問題。
如果每一次在不同網(wǎng)絡(luò)環(huán)境間切換,都需要更改代碼,然而重新打包,那未免有點(diǎn)低效。下面是我的實(shí)踐探索,看網(wǎng)上很多人都是根據(jù)buildType來切換網(wǎng)絡(luò)環(huán)境,感覺有點(diǎn)不好,因?yàn)榫W(wǎng)絡(luò)環(huán)境可能很多種,而buildType我們一般是2種,而且,不同網(wǎng)絡(luò)環(huán)境的包最好能同時(shí)安裝在手機(jī)上,以便我們調(diào)試。最好,我一看這個(gè)包的名稱和圖標(biāo),就能知道這是什么環(huán)境的包。
一、概述
1.多版本
基于buildTypes
(1)debug:調(diào)試版本,無混淆
(2)release:發(fā)布版本,有混淆、壓縮
2.多環(huán)境
基于productFlavors
(1)develop:開發(fā)環(huán)境,開發(fā)和自測(cè)時(shí)使用
(2)check:測(cè)試環(huán)境,克隆一份生產(chǎn)環(huán)境的配置,在這里測(cè)試通過后,再發(fā)布到生產(chǎn)環(huán)境。
之所以沒命名為test是因?yàn)樵趃radle編譯時(shí):ProductFlavor names cannot start with 'test'
(3)product:生產(chǎn)環(huán)境,正式提供服務(wù)的。
3.多渠道
基于Android新的應(yīng)用簽名方案APK Signature Scheme v2中的APK Signing Block區(qū)塊
我這里使用的是美團(tuán)封裝的Walle庫。使用Walle庫請(qǐng)確保你的Android Gradle 插件版本在2.2.0以上。
為什么不直接使用productFlavors來打包多渠道?因?yàn)閜roductFlavors打多渠道包太慢了,打30個(gè)包差不多十幾分鐘,無法忍受!
為什么不使用美團(tuán)之前基于META-INF進(jìn)行渠道標(biāo)識(shí)的方案?因?yàn)锳ndroid7.0之后的這種黑科技已經(jīng)失效了!
二、示例
1、配置build.gradle
(1) 在位于項(xiàng)目的根目錄 build.gradle 文件中添加Walle Gradle插件的依賴, 如下:
buildscript { dependencies { classpath 'com.android.tools.build:gradle:2.2.3' classpath 'com.meituan.android.walle:plugin:1.0.3' } }
(2) 在當(dāng)前App的 build.gradle 文件中apply這個(gè)插件,并添加上用于讀取渠道號(hào)的aar
apply plugin: 'com.android.application' apply plugin: 'walle' android { compileSdkVersion 25 buildToolsVersion "25.0.2" defaultConfig { minSdkVersion 15 targetSdkVersion 25 versionCode 1 versionName "1.0" testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner" } signingConfigs { release { keyAlias KEY_ALIAS keyPassword KEY_PASSWORD storeFile rootProject.file(KEYSTORE_FILE) storePassword KEYSTORE_PASSWORD } } buildTypes { //調(diào)試版本,無混淆 debug { minifyEnabled false signingConfig signingConfigs.release } //發(fā)布版本,有混淆 release { minifyEnabled true zipAlignEnabled true shrinkResources true signingConfig signingConfigs.release proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro' } } productFlavors { //開發(fā)環(huán)境 develop { buildConfigField "int", "ENV_TYPE", "1" applicationId 'om.soubu.walledemo.develop' manifestPlaceholders = [ app_name: "開-WalleDemo", app_icon: "@drawable/icon_develop" ] } //測(cè)試環(huán)境 check { buildConfigField "int", "ENV_TYPE", "2" applicationId 'om.soubu.walledemo.check' manifestPlaceholders = [ app_name: "測(cè)-WalleDemo", app_icon: "@drawable/icon_check" ] } //生產(chǎn)環(huán)境 product { buildConfigField "int", "ENV_TYPE", "3" applicationId 'com.soubu.walledemo.product' manifestPlaceholders = [ app_name: "WalleDemo", app_icon: "@drawable/icon_product" ] } } } dependencies { compile fileTree(dir: 'libs', include: ['*.jar']) androidTestCompile('com.android.support.test.espresso:espresso-core:2.2.2', { exclude group: 'com.android.support', module: 'support-annotations' }) compile 'com.android.support:appcompat-v7:25.1.0' testCompile 'junit:junit:4.12' compile 'com.meituan.android.walle:library:1.0.3' }
(3) 這里,我根據(jù)不同的環(huán)境生成了不同包名的apk,方便在手機(jī)上同時(shí)安裝多個(gè)環(huán)境的應(yīng)用。為了讓gradle動(dòng)態(tài)更改apk的名稱和圖標(biāo),我們需要在manifest文件中使用
{app_name}等占位符
<?xml%20version="1.0"%20encoding="utf-8"?> <manifest%20xmlns:android="http://schemas.android.com/apk/res/android" %20package="com.soubu.walledemo"> %20<application %20android:allowBackup="true" %20android:icon="${app_icon}" %20android:label="${app_name}" %20android:supportsRtl="true" %20android:theme="@style/AppTheme"> %20<activity%20android:name=".MainActivity"> %20<intent-filter> %20<action%20android:name="android.intent.action.MAIN"/> %20<category%20android:name="android.intent.category.LAUNCHER"/> %20</intent-filter> %20</activity> %20</application> </manifest>
(4)%20在代碼中獲取多渠道信息
String%20channel%20=%20WalleChannelReader.getChannel(getApplicationContext());
(5)%20在代碼中獲取多環(huán)境信息
int%20envType%20=%20BuildConfig.ENV_TYPE;
這里的BuildConfig是由gradle動(dòng)態(tài)生成的:
package%20com.soubu.walledemo; public%20final%20class%20BuildConfig%20{ %20public%20static%20final%20boolean%20DEBUG%20=%20Boolean.parseBoolean("true"); %20public%20static%20final%20String%20APPLICATION_ID%20=%20"om.soubu.walledemo.develop"; %20public%20static%20final%20String%20BUILD_TYPE%20=%20"debug"; %20public%20static%20final%20String%20FLAVOR%20=%20"develop"; %20public%20static%20final%20int%20VERSION_CODE%20=%201; %20public%20static%20final%20String%20VERSION_NAME%20=%20"1.0"; %20//%20Fields%20from%20product%20flavor:%20develop %20public%20static%20final%20int%20ENV_TYPE%20=%201; }
而ENV_TYPE這個(gè)字段其實(shí)就來自于我們的build.gradle:
%20productFlavors%20{ %20//開發(fā)環(huán)境 %20develop%20{ %20buildConfigField%20"int",%20"ENV_TYPE",%20"1" %20applicationId%20'om.soubu.walledemo.develop' %20manifestPlaceholders%20=%20[ %20app_name:%20"開-WalleDemo", %20app_icon:%20"@drawable/icon_develop" %20] %20} %20{
這里我們最好定義一個(gè)常量類區(qū)分這些環(huán)境的類型:
public%20class%20EnvType%20{ %20public%20static%20final%20int%20DEVELOP%20=%201;//開發(fā)環(huán)境 %20public%20static%20final%20int%20CHECK%20=%202;//測(cè)試環(huán)境 %20public%20static%20final%20int%20PRODUCT%20=%203;//正式環(huán)境 }
2、打包多環(huán)境
這里我們直接執(zhí)行assemble命令,打包所有的buildType*productFlavors
或者使用命令行也可以:
gradle assemble
執(zhí)行結(jié)果:26秒搞定6個(gè)包:2個(gè)版本*3個(gè)環(huán)境


這里我們可以看到debug包都是1.4M,而release包都是0.7M,顯然,我們的混淆和壓縮配置是生效了的,雖然這里我并沒寫混淆規(guī)則


我們分別安裝3個(gè)環(huán)境的包到自己的手機(jī)上:

看三個(gè)包的名稱和圖標(biāo)都不一樣,顯然我們之前在manifest文件中配置的占位符生效了。
然后我們點(diǎn)進(jìn)去分別看看這3個(gè)app的區(qū)別:



這樣,我們就可以在代碼中,根據(jù)環(huán)境字段envType的不同,來選擇不同的網(wǎng)絡(luò)環(huán)境了。
界面的代碼如下:
public class MainActivity extends AppCompatActivity { @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); TextView tvEnv = (TextView) findViewById(R.id.tv_env); TextView tvChannel = (TextView) findViewById(R.id.tv_channel); TextView tvPackage = (TextView) findViewById(R.id.tv_package); String channel = WalleChannelReader.getChannel(this.getApplicationContext()); int envType = BuildConfig.ENV_TYPE; String packageName = getPackageName(); switch (envType) { case EnvType.DEVELOP: tvEnv.setText("envType=" + "開發(fā)環(huán)境"); break; case EnvType.CHECK: tvEnv.setText("envType=" + "測(cè)試環(huán)境"); break; case EnvType.PRODUCT: tvEnv.setText("envType=" + "生產(chǎn)環(huán)境"); break; } tvChannel.setText("channel=" + channel); tvPackage.setText("package=" + packageName); } }
3、打包多渠道
在Project的根目錄下新建channel文件:
anzhi #安智 baidu #百度 huawei #華為 oppo #oppo wdj #豌豆莢 xiaomi #小米 yyb #應(yīng)用寶
執(zhí)行g(shù)radle命令:
(1) 打包文件內(nèi)的渠道包
gradle assembleProductRelease -PchannelFile=channel
(2) 打包自定義數(shù)組內(nèi)的渠道包
gradle assembleProductRelease -PchannelList=qihu,vivo,lenovo
關(guān)于Walle庫的更多使用:詳見Github-walle
運(yùn)行結(jié)果:17秒搞定8個(gè)包:1個(gè)默認(rèn)包+7個(gè)渠道包


最后,奉上源碼:WalleDemo
常見問題
1、找不到簽名文件的配置?

汗,因?yàn)槲业腄emo中并沒有上傳我的jks文件,你可以添加自己的jks文件,然后在gradle.properties里面配置好簽名文件的密碼即可
在gradle.properties添加簽名文件的配置key-value

在build.gradle中引用配置的key

2、develop、check、product,如果直接run代碼,怎么設(shè)置默認(rèn)的環(huán)境?
點(diǎn)擊查看AndroidStudio左下角的BuildVariants,然后選擇設(shè)置默認(rèn)的run環(huán)境即可。
BuildVariants= buildTypes* productFlavors
