Android多渠道打包

使用 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
            ......
        }
    }
}

或者


generate_signed_apk

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)

使用时需要定义多渠道版本文件


channel.png
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
shell.png
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,362评论 5 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,330评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,247评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,560评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,580评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,569评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,929评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,587评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,840评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,596评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,678评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,366评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,945评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,929评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,165评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 43,271评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,403评论 2 342

推荐阅读更多精彩内容

  • Android多渠道打包 概述 每当发新版本时,Android客户端会被分发到各个应用市场,比如豌豆荚,360手机...
    砺雪凝霜阅读 2,111评论 2 11
  • 目录一、Python打包及优化(美团多渠道打包)二、Gradle打包三、其他打包方案:修改Zip文件的commen...
    守望君阅读 5,675评论 4 17
  • ###多渠道打包概念 >* 发布到不同渠道上的apk,标记不同的渠道,目的是为了统计该渠道的下载量,留存率等等数据...
    未聞椛洺阅读 593评论 0 0
  • 我们做Android用户级应用开发的时候都要考虑这样的问题,目前的应用市场有很多,我们的安装包是通过哪个渠道进入用...
    尹star阅读 8,744评论 11 26
  • 每次中午吃饭总会和技术同学聊天。当做 iOS 开发的做安卓开发的人员在一起的时候,他们中间又多了一个话题:iOS ...
    OneAPM_Official阅读 2,000评论 1 6