关闭

解决apk多渠道对应不同后台的问题

713人阅读 评论(0) 收藏 举报

需求

在项目中碰到一个问题,就是一个android项目有多个渠道,其中每个渠道对应的后台地址不相同,要部署这么一个庞大的系统,android studio多渠道打包能解决这个问题,但是维护这么一个庞大的系统升级就会显得无力了,于是讨论了一些方案

方案

方案1:最原始的方法,针对每个项目都做apk编译,修改好后台路径后编译生成apk,维护一次上传一次
方案2:利用android studio多渠道打包方式,一次生成N多个不同渠道的apk,升级一次所有的地方都要重新上传
方案3:解放生产力,建立一个服务端统一管理所有的apk,下载时重打包生成对应的不同的后台路径的apk
针对上面三种不同的方案,第一种维护起来相当麻烦,每次都需要开发人员编译不同的版本,上传到不同的后台服务器中,维护起来相当麻烦,首先否定;第二种方案还是要程序员手工生成不同的版本,而且维护升级也是比较麻烦;第三种方案重打包生成apk省去了人工出错的几率,有一定的便捷性,所以这里采取第三种方案

这里第三种方案最初想到的是用apktool重打包,考虑到apktool打包性能问题,所以当时就否决了这种方案。于是搜索网上资源查看到美团重打包加入空文件的多渠道打包方案,觉得很方便,想采用这种方案,但最终发现了一个更好的方案,于是就采用了这种方案
文章地址:
http://pingguohe.net/2016/03/21/Dynimac-write-infomation-into-apk.html#rd?sukey=3903d1d3b699c208760b00c7ef24e8d44b232e0b0ae8a7306f175fc470886375681de7ea015a80ac50e81680056e2bca

原理

android使用的apk包的压缩方式是zip,与zip有相同的文件结构,在zip的Central directory file header中包含一个File comment区域,可以存放一些数据。File comment是zip文件如果可以正确的修改这个部分,就可以在不破坏压缩包、不用重新打包的的前提下快速的给apk文件写入自己想要的数据。
comment是在Central directory file header末尾储存的,可以将数据直接写在这里,下是header末尾的结构。 

由于数据是不确定的,我们无法知道comment的长度,从表中可以看到zip定义comment的长度的位置在comment之前,所以无法从zip中直接获取comment的长度。这里我们需要自定义comment的长度,在自定义comment内容的后面添加一个区域储存comment的长度,结构如下图。
这里可以将一个固定的结构写在comment中,然后根据自定义的长度分区获取每个部分的内容,添加版本quda。

实现

1.将数据写入comment

这一部分可以在本地进行,需要定义一个长度为2的byte[]来储存comment的长度,直接使用Java的api就可以把comment和comment的长度写到apk的末尾,代码如下。

public static void writeApk(File file, String comment) {
    ZipFile zipFile = null;
    ByteArrayOutputStream outputStream = null;
    RandomAccessFile accessFile = null;
    try {
        zipFile = new ZipFile(file);
        String zipComment = zipFile.getComment();
        if (zipComment != null) {
            return;
        }

        byte[] byteComment = comment.getBytes();
        outputStream = new ByteArrayOutputStream();

        outputStream.write(byteComment);
        outputStream.write(short2Stream((short) byteComment.length));

        byte[] data = outputStream.toByteArray();

        accessFile = new RandomAccessFile(file, "rw");
        accessFile.seek(file.length() - 2);
        accessFile.write(short2Stream((short) data.length));
        accessFile.write(data);
    } catch (IOException e) {
        e.printStackTrace();
    } finally {
        try {
            if (zipFile != null) {
                zipFile.close();
            }
            if (outputStream != null) {
                outputStream.close();
            }
            if (accessFile != null) {
                accessFile.close();
            }
        } catch (Exception e) {

        }

    }
}
         
	private static byte[] short2Stream(short data) {
		ByteBuffer buffer = ByteBuffer.allocate(2);
		buffer.order(ByteOrder.LITTLE_ENDIAN);
		buffer.putShort(data);
		buffer.flip();
		return buffer.array();
	}

2.读取apk包中的comment数据

首先获取apk的路径,通过context中的getPackageCodePath()方法就可以获取,代码如下。

public static String getPackagePath(Context context) {
    if (context != null) {
        return context.getPackageCodePath();
    }
    return null;
}

获取路径之后就可以读取comment的内容了,这里不能直接使用ZipFile中的getComment()方法直接获取comment,因为这个方法是Java7中的方法,在android4.4之前是不支持Java7的,所以我们需要自己去读取apk文件中的comment。首先根据之前自定义的结构,先读取写在最后的comment的长度,根据这个长度,才可以获取真正comment的内容,代码如下。

public static String readApk(File file) {
    byte[] bytes = null;
    try {
        RandomAccessFile accessFile = new RandomAccessFile(file, "r");
        long index = accessFile.length();

        bytes = new byte[2];
        index = index - bytes.length;
        accessFile.seek(index);
        accessFile.readFully(bytes);

        int contentLength = stream2Short(bytes, 0);

        bytes = new byte[contentLength];
        index = index - bytes.length;
        accessFile.seek(index);
        accessFile.readFully(bytes);

        return new String(bytes, "utf-8");
    } catch (FileNotFoundException e) {
        e.printStackTrace();
    } catch (IOException e) {
        e.printStackTrace();
    }
    return null;
}

这里的stream2Short()和short2Stream()参考了MultiChannelPackageTool中的方法。

测试

在生成apk后,调用下面的代码写入我们想要的数据,

File file = new File("/Users/zhaolin/app-debug.apk");
writeApk(file, "test comment");

安装这个apk之后运行,让comment显示在屏幕上,运行结果如下。 
运行结果符合预期,安装包也没有被破坏,可以正常安装。

结论

  • 通过修改comment将数据传递给APP的方案是可行的,由于是修改apk自有的数据,并不会对apk造成破坏,修改后可以正常安装。
  • 这种方案不用重新打包apk,并且在服务端只是写文件的操作,效率很高,可以适用于动态生成apk的场景。
  • 可以通过这个版本号获取本客户端生成的渠道信息,用后台统一维护,做到运维实施便捷化。
最后感谢原作者共享技术,使我们遇到问题能快速解决
0
0

查看评论
* 以上用户言论只代表其个人观点,不代表CSDN网站的观点或立场
    个人资料
    • 访问:15328次
    • 积分:249
    • 等级:
    • 排名:千里之外
    • 原创:4篇
    • 转载:28篇
    • 译文:1篇
    • 评论:0条
    文章分类