[关闭]
@linux1s1s 2016-06-17T09:19:16.000000Z 字数 3765 阅读 3365

Android 多渠道打包方案

AndroidBuild


常规Build

我们先来回顾一下通过Ant或者Gradle进行多渠道批量打包,通常是在AndroidManifest中配置:

  1. <meta-data android:name="CHANNEL" android:value="xxx" />

meta-data通过配置value来动态改变渠道名称,然后我们可以在代码中这样去获取Channel

  1. private String getChannelNameFromManifest(){
  2. try {
  3. return (String) mContext.getPackageManager().getApplicationInfo(mContext.getPackageName(), PackageManager.GET_META_DATA).metaData.get("CHANNEL");
  4. } catch (NameNotFoundException e) {
  5. e.printStackTrace();
  6. }
  7. }

Ant 在多渠道包上配置起来相对麻烦一点,如果你感兴趣的话可以参考这个博客 谦虚的天下-《App自动化之使用Ant编译项目多渠道打包》
而Gradle作为Android官方构建工具,批量打包自然会相对简单一点,就像下面这样简单配置一下即可:

  1. /**
  2. * -------------------------------------------------------
  3. * Flavors
  4. * --------------------------------------------------------
  5. */
  6. productFlavors {
  7. intl {
  8. applicationId "com.xxx.x1"
  9. versionName createVersionName(versionNameInit)
  10. manifestPlaceholders = [deeplinkScheme: "x1"]
  11. }
  12. gloav {
  13. applicationId "com.xxx.x2"
  14. versionName createVersionName(versionNameInit)
  15. manifestPlaceholders = [deeplinkScheme: "x2"]
  16. }
  17. }

配置可以如此简单,但是批量打包还是绕不过从新构建这个过程,用一句通俗的话说就是,如果我需要1000个渠道包,每个渠道包需要5分钟,那么批量完成这些包需要5000分钟,Android在开发Gradle的时候可能完全想不到这个问题,因为除了天朝有这么多渠道市场外,其他国家都是望其项背,啰嗦几句后,咱们还是得面对现实,解决这个耗时的打包问题。

这里从最简单的情况说起,如果这么多渠道仅仅是区分不同的Channel,其他都相同。如果还有别的细微区别,这里暂且不考虑,话说回来,即使是这些细微的区别可能也仅仅是个别渠道,所以最笨的解决方法是从新构建这些个别渠道,其他都可以通过捷径来解决。

这个捷径是什么呢,当然是避免从新构建Build,如果不从新构建那么如果做到区分不同的渠道呢?

优化Build

我们先来看一下APK的构造

此处输入图片的描述

这个压缩目录中META-INF目录中的文件原则上是不参与签名的,所以我们可以通过在该文件中放入Channel标示而逃过从新构建,这样我们就可以直接通过文件操作来完成批量打包,而文件的复制解压压缩操作所消耗的时间和从新构建消耗的时间相比,简直可以忽略不计,上面就是整个打包方案的核心思想,接下来我们来实施上面的核心思想。

实际操作

在META-INF这个目录中添加一个以xxx_channelName_channelId命名的空文件,以xiaomi渠道举个例子如下所示:

此处输入图片的描述

然后再将解压后并添加相应文件的目录重新压缩为.apk格式的文件即可,上面的工作完成了,接下来就是批量添加文件过程了,这里给出Python脚本以供参考

Python脚本

  1. #!/usr/bin/python
  2. # coding=utf-8
  3. import zipfile
  4. import shutil
  5. import os
  6. # 空文件 便于写入此空文件到apk包中作为channel文件
  7. src_empty_file = 'xxxxx.txt'
  8. #创建一个空文件(不存在则创建)
  9. with open(src_empty_file, 'w') as f:
  10. pass
  11. # 获取当前目录中所有的apk源包
  12. src_apks = []
  13. # python3 : os.listdir()即可,这里使用兼容Python2的os.listdir('.')
  14. for file in os.listdir('.'):
  15. if os.path.isfile(file):
  16. extension = os.path.splitext(file)[1][1:]
  17. if extension in 'apk':
  18. src_apks.append(file)
  19. # 获取渠道列表
  20. channel_file = 'channelConfig.txt'
  21. with open(channel_file) as f:
  22. for src_apk in src_apks:
  23. # file name (with extension)
  24. src_apk_file_name = os.path.basename(src_apk)
  25. # 分割文件名与后缀
  26. temp_list = os.path.splitext(src_apk_file_name)
  27. # name without extension
  28. src_apk_name = temp_list[0]
  29. # 后缀名,包含. 例如: ".apk "
  30. src_apk_extension = temp_list[1]
  31. # 创建生成目录,与文件名相关
  32. output_dir = 'output_' + src_apk_name + '/'
  33. # 目录不存在则创建
  34. if not os.path.exists(output_dir):
  35. os.mkdir(output_dir)
  36. # 遍历渠道号并创建对应渠道号的apk文件
  37. for line in f.readlines():
  38. # 获取当前渠道号,因为从渠道文件中获得带有\n,所有strip一下
  39. target_channel = line.strip()
  40. # 获取渠道号,从中分离出渠道name和渠道id
  41. target_channel_split = target_channel.split('_')
  42. target_channel_name = target_channel_split[0]
  43. #target_channel_id = target_channel_split[1]
  44. # 拼接对应渠道号的apk
  45. target_apk = output_dir + src_apk_name + '-release-' + target_channel_name + src_apk_extension
  46. # 拷贝建立新apk
  47. shutil.copy(src_apk, target_apk)
  48. # zip获取新建立的apk文件
  49. zipped = zipfile.ZipFile(target_apk, 'a', zipfile.ZIP_DEFLATED)
  50. # 初始化渠道信息
  51. empty_channel_file = "META-INF/xxxx_{channel}".format(channel = target_channel)
  52. # 写入渠道信息
  53. zipped.write(src_empty_file, empty_channel_file)
  54. # 关闭zip流
  55. zipped.close()

最后在代码中获取这个channel标示即可。

  1. private String getChannelFromApk(String channelKey) {
  2. ApplicationInfo appinfo = mContext.getApplicationInfo();
  3. String sourceDir = appinfo.sourceDir;
  4. String key = "META-INF/" + channelKey;
  5. String ret = "";
  6. ZipFile zipfile = null;
  7. try {
  8. zipfile = new ZipFile(sourceDir);
  9. Enumeration<?> entries = zipfile.entries();
  10. while (entries.hasMoreElements()) {
  11. ZipEntry entry = ((ZipEntry) entries.nextElement());
  12. String entryName = entry.getName();
  13. if (entryName.startsWith(key)) {
  14. ret = entryName;
  15. break;
  16. }
  17. }
  18. } catch (IOException e) {
  19. e.printStackTrace();
  20. } finally {
  21. if (zipfile != null) {
  22. try {
  23. zipfile.close();
  24. } catch (IOException e) {
  25. e.printStackTrace();
  26. }
  27. }
  28. }
  29. String[] split = ret.split("_");
  30. String channel = "";
  31. if (split != null && split.length >= 2) {
  32. channel = ret.substring(split[0].length() + 1);
  33. }
  34. return channel;
  35. }

基本上到这里就OK了,如果问题,欢迎讨论。

添加新批注
在作者公开此批注前,只有你和作者可见。
回复批注