安装包大小优化

1.常规的安装包优化

  • 图片压缩
  • 资源动态加载
  • Lint工具
  • 极限压缩
  • Proguard 混淆

2.资源文件再压缩

  • 思路
  • 工具

安装包优化宗旨:
一个字:删!!删不了就尽量小。

1.常规的安装包的优化:

1.1.图片压缩

(1)图片:apk里面的资源图片 压缩图片
(2)svg图片:一些图片的描述,牺牲CPU的计算能力的,节省空间。

svg使用的原则:简单的图标。

(3)webp:谷歌现在非常提倡的使用。保存图片比较小。
VP8派生而来的。webp的无损压缩比PNG文件小45%左右,即使PNG进过其他的压缩工具压缩后,任然可以减小到PNG的28%。

使用公司:Facebook在用、腾讯、淘宝。
缺点:加载相比于PNG要慢很多。 但是现在手机配置比较高。
转换工具:http://isparta.github.io/

1.2.资源动态加载:
  • 比如:emoji表情、换肤
  • 动态下载的资源。
  • 一些模块的插件化动态添加。

1.3.Lint工具

目的是让APP瘦身,提出一些建议优化的点:

  • 1)检测没有用的布局;(建议 删除)
  • 2)未使用到的资源; (比如 图片 ---删除)
  • 3)建议String.xml有一些没有用到的字符;

1.4.极限压缩

7zZip工具的使用。


image.png

图片压缩
通过色彩深度,透明通道的压缩,不明显降低图片质量情况下减小图片大小。色彩深度计算机图形学领域表示在位图或者视频帧缓冲区中储存1像素的颜色所用的位数,它也称为位/像素(bpp)。色彩深度越高,可用的颜色就越多。
PNG图片优化压缩的文章: http://www.wufangbo.com/png-image-optimization/

1.5.Proguard 混淆。

让apk变小。为什么?

  • 1)可以删除注释和不用的代码。
  • 2)将java文件名改成短名a.java,b.java
  • 3)方法名等 CommonUtil.getDisplayMetrix();——>a.a()

2.资源文件再压缩

在常规的安装包的优化之外继续压缩---资源文件再压缩

系统编译完成apk文件以后,R文件中映射关系:

res/drawable/ic_launcher.png ----- > 0x7f020000

2.1思路:

再做“混淆”:要实现将res/drawable/ic_launcher.png图片改成a.png

  • drawable文件的名字
  • String文件的名字
  • layout的名字
    比如:R.string.description--->R.string.a
    res/drawable/ic_launcher.png图片改成a.png

还可以更加夸张:

res/drawable--->r/d
res/value-->r/v
res/drawable/ic_launcher.png图片改成r/d/a.png

读取resources.arsc二进制文件,然后修改某一段一段的字节。
有一段叫做:res/drawable/ic_launcher.png 在自己数组当中的第800位-810位
将这一段第800位-810位替换成改成r/d/a.png 的字节码。

resources_arsc二进制表:


resources_arsc二进制表.png
public class ArscTest {
    public static void main(String[] args){
        ArscTest arscTest = new ArscTest();
        try {
//          arscTest.readInputStream("C:/Users/luoding/workspace_test/arscTest/input.apk");
//          arscTest.readInputStream("C:/Users/luoding/workspace_test/arscTest/Lsn10SearchView.apk");
            arscTest.readInputStream("C:/Users/luoding/workspace_test/arscTest/dn_jobschduler.apk");
        } catch (IOException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }
    }
    
    @SuppressWarnings("resource")
    public void readInputStream(String path) throws IOException{
        ZipFile zipFile = new ZipFile(path);
        InputStream inputStream = zipFile.getInputStream(new ZipEntry(""+"resources.arsc"));
        LEDataInputStream leDataInputStream = new LEDataInputStream(inputStream);
        //Resource Table 头
        short type = leDataInputStream.readShort();
        leDataInputStream.skipBytes(2);
        leDataInputStream.readInt();
        int packageNum = leDataInputStream.readInt();
        System.out.println("num of package:"+packageNum);
        
        //StringPool块
        int got =leDataInputStream.readInt();
        //块大小
        int chunkSize = leDataInputStream.readInt();
        //字符串数量
        int stringCount = leDataInputStream.readInt();
        //style数量
        int styleCount = leDataInputStream.readInt();
        //标记
        int flags = leDataInputStream.readInt();
        //字符串起始位置
        int stringsOffset = leDataInputStream.readInt();
        //style起始位置
        int stylesOffset = leDataInputStream.readInt();
        int[] array = new int[stringCount];
        for (int i = 0; i < stringCount; ++i){
            array[i] = leDataInputStream.readInt();
        }
        if (styleCount != 0) {
            for (int i = 0; i < styleCount; ++i)
                array[i] = leDataInputStream.readInt();
        }
        //字符串长度
        int size = ((stylesOffset == 0) ? chunkSize : stylesOffset) - stringsOffset;
        byte[] m_strings = new byte[size];
        StringBuffer ss = new StringBuffer();
        leDataInputStream.readFully(m_strings); 
        for(int i = 0;i<m_strings.length;i++){
            //(通过打开resources.arsc看到一些乱码 猜得出字符都是ASCII码)
            char c = (char) m_strings[i];
            ss.append(c);
        }
        System.out.println(ss.toString());
        if (stylesOffset != 0) {
            size = chunkSize - stylesOffset;
            if (size % 4 != 0)
                throw new IOException("Style data size is not multiple of 4 (" + size + ").");

            for (int i = 0; i < size / 4; ++i)
                leDataInputStream.readInt();
        }
        //nextChunk
        leDataInputStream.readShort();
        leDataInputStream.skipBytes(2);
        leDataInputStream.readInt();
        
        int id = (byte) leDataInputStream.readInt();
        StringBuilder sb = new StringBuilder(16);
        int length = 256;
        while (length-- != 0) {
              short ch = leDataInputStream.readShort();
              if (ch == 0)
                break;

              sb.append((char)ch);
            }
        System.out.println("pacakgeName:"+sb.toString());
    }
    
    
    
}..................

aapt打包过程:


aapt.png

2.2.工具

什么是Zipalign?什么是Zipaligned?
http://bbs.ihei5.com/thread-171596-1-1.html

main函数运行配置:
在项目上右键-->Run as--->RunConfiguration
选择Arguments
在Program arguments里面填写如下:

input.apk -config config.xml -out outapkdir/haha -7zip 7za.exe

或者
input.apk -config config.xml -7zip 7za.exe

args参数:
Lsn10SearchView.apk -config config.xml -7zip 7za.exe -out xxx/hehe -mapping xxx/yyy.txt

1.12M
1.03M
920K

推荐阅读更多精彩内容