使用
Android studio
自帶的打包工具通過productFlavors
來打多渠道包,效率太低,每次只變更了一個渠道名稱,卻要重頭打包編譯,後來看到 美團多渠道方案,確實很方便,參考別人的程式碼整理了份指令碼,在此記錄一下; 系統: mac 10.12 python: 3.x
要求:
1. 只使用使用v1簽名方式:
P.S. 若使用v2簽名的,可以參考 這篇 ;
android{
signingConfigs {
release {
keyAlias '******'
keyPassword '******'
storeFile file('***.jks')
storePassword '******'
v2SigningEnabled false // 禁用v2簽名
}
}
buildTypes {
release {
signingConfig signingConfigs.release
......
}
}
}
複製程式碼
或者
2. 第三方庫的渠道名稱可通過程式碼來設定
如我專案中使用的talkingdata,就可以先去查詢 META_INF/
中的渠道檔案,若有則使用其定義的渠道名來初始化:
private void initTalkingData() {
String channel = getChannel(this);
String tdAppId = CommonUtils.getMetaValue(this, "TD_APP_ID");
if (TextUtils.isEmpty(channel) || TextUtils.isEmpty(tdAppId)) {
TCAgent.init(this);
} else {
TCAgent.init(this, tdAppId, channel);
}
TCAgent.setReportUncaughtExceptions(true);
}
複製程式碼
使用
/**
* 獲取meta_data中指定的渠道號
* 檔案為: META-INF/tdchannel_{channelName}
*/
public static String getChannel(Context context) {
ApplicationInfo appinfo = context.getApplicationInfo();
String sourceDir = appinfo.sourceDir;
String ret = "";
ZipFile zipfile = null;
try {
zipfile = new ZipFile(sourceDir);
Enumeration<?> entries = zipfile.entries();
while (entries.hasMoreElements()) {
ZipEntry entry = ((ZipEntry) entries.nextElement());
String entryName = entry.getName();
// 檔名與python指令碼定義的相匹配即可
if (entryName.startsWith("META-INF/tdchannel")) {
ret = entryName;
break;
}
}
} catch (Exception e) {
e.printStackTrace();
} finally {
if (zipfile != null) {
try {
zipfile.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
String[] split = ret.split("_");
if (split.length >= 2) {
return ret.substring(split[0].length() + 1);
} else {
return "";
}
}
複製程式碼
python指令碼
把下面的程式碼儲存為 multi_channel.py
檔案
#!/usr/bin/env python3
# -*- coding: utf-8 -*-
'''
根據美團多渠道打包方式,自動生成多渠道目錄
要求apk是使用V1加密方式打包的;
python3 multi_channel -s srcApkPath -v versionCode
python3 multi_channel -srcFile=srcApkPath --version=versionCode
'''
import os
import shutil
import zipfile
import time
import sys
import getopt
startTime = time.time()
prefixInfo = "release"
srcApk = "./channelApk.apk"
version = "" # 版本號
channelFilePath = "./channel" # 渠道配置檔案路徑,每行定義一個渠道
toolInfo = '''參考美團多渠道打包方案1的打包工具;
預設使用當前目錄下 "channel" 檔案中定義的渠道資訊,每行一個渠道名稱,可通過 -c 引數來指定渠道檔案路徑;
要求apk使用的是v1簽名方式,若使用v2則本工具無效;
python3 multi_channel -s srcApkPath -v 1.7 -p demo -c ./channel
-s --srcFile : 新增一個源apk,會依據該apk生成多渠道apk,並儲存於 "./channelApk/" 中;
-v --version : 可選, 給生成的apk名稱新增一個版本號,會自動新增字首 _v{version},如 demo_v1.7.apk;
-p --prefix : 可選, 給生成的apk名稱新增一個字首資訊,預設為"release"
-c --channel : 定義要生成的渠道包資訊,每行定義一個渠道名稱,會依次生成對應的渠道包'''
opts, args = getopt.getopt(sys.argv[1:], "hs:v:p:c:", ["help", "srcFile=", "version=", "prefix=", "channel="])
for name, value in opts:
if name in ("-s", "--srcFile"): # 原始檔名稱
srcApk = value
elif name in ("-v", "--version"): # 版本號
version = "_v%s" % value
elif name in ("-p", "--prefix"): # apk名稱字首資訊
prefixInfo = value
elif name in ("-c", "--channel"): # 多渠道配置檔案
channelFilePath = value
elif name in ("-h", "--help"):
print(toolInfo)
exit()
print("srcApk = %s , version = %s" % (srcApk, version))
isApkExist = os.path.exists(srcApk)
if not isApkExist or not os.path.isfile(srcApk):
print("%s 源apk檔案不存在,請重試" % srcApk)
exit()
if not os.path.exists(channelFilePath) or not os.path.isfile(channelFilePath):
print("%s channel渠道檔案不存在或者不是有效的file,請檢查後重試" % channelFilePath)
exit()
pkgPath = os.path.join(os.getcwd(), "channelApk") # 生成的多渠道apk存放的目錄
print("生成的apk會被存放於 %s" % pkgPath)
isPathExist = os.path.exists(pkgPath)
isDir = os.path.isdir(pkgPath)
if not isPathExist or not isDir:
os.makedirs(pkgPath)
f = open(channelFilePath, 'r', encoding='utf-8')
for line in f:
channel_name = line.strip('\n')
# print("當前正在生成渠道包: %s" % channel_name)
channelPath = pkgPath + "/{prefix}_{channel}{version}.apk".format(prefix=prefixInfo, channel=channel_name,
version=version)
shutil.copy(srcApk, channelPath)
zipped = zipfile.ZipFile(channelPath, 'a', zipfile.ZIP_DEFLATED)
empty_channel_file = "META-INF/tdchannel_{channel}".format(channel=channel_name)
# zipped.write("empty", empty_channel_file) # 使用這種方式需要在當前目錄下存在empty檔案
zipped.writestr(empty_channel_file, data=channel_name)
diff = time.time() - startTime
print("耗時: %s" % diff)
複製程式碼
使用時需要定義多渠道版本檔案
python3 multi_channel -s ./SrcApk.apk -v 1.0 -p demo -c ./channel
// 就會在 `./channelApk/` 中生成多渠道apk包,apk檔名類似: demo_baidu_v1.0.apk
// 若channel 檔案位於當前目錄下,則可省略 -c ./channel ,即:
python3 multi_channel -s ./SrcApk.apk -v 1.0 -p demo
複製程式碼
當然,也可以給 multi_channel.py
新增執行許可權,並將其所在目錄新增到PATH
環境變數中,就不需要使用 python3
命令來執行了;
chmod a+x multi_channel.py //新增可執行許可權
vi ~/.bash_profile
// 新增 multi_channel.py 檔案所在目錄到 PATH
export PATH = ****:$PATH // ****表示python指令碼所在目錄路徑
source ~/.bash_profile
multi_channel.py -s SrcApk.apk -c channel
複製程式碼