# apk-shrink **Repository Path**: cankingapp/apk-shrink ## Basic Information - **Project Name**: apk-shrink - **Description**: Apk shrink tool Demo. - **Primary Language**: Java - **License**: Apache-2.0 - **Default Branch**: master - **Homepage**: None - **GVP Project**: No ## Statistics - **Stars**: 0 - **Forks**: 0 - **Created**: 2017-11-28 - **Last Updated**: 2020-12-19 ## Categories & Tags **Categories**: Uncategorized **Tags**: None ## README # APK瘦身实战 资源自定义7z压缩 项目开发中,随着业务的增长,常常需要在apk编译阶段对包代码或是资源做一定的自定义修改,比如热修复,插件生成,无埋点统计,渠道包生成等等。 但是公司项目业务开发人员基本上都很少接触到相关技术,这里以学习的态度,实现一套用7zip压缩apk资源文件的gradle插件。 APK瘦身在行业内已经有很多成熟的开源技术体现方案,如美团[Android App包瘦身优化实践](https://tech.meituan.com/android-shrink-overall-solution.html)这篇博客中详细的说明, 这里我们从资源压缩入手,用7z工具,实现一套自己的资源压缩gradle插件。 ## Gradle 插件简单介绍 一般简单的逻辑可以写在build.gradle文件中,但是本着便于管理及重用,可以把插件独立为项目。 独立的gradle插件编译后,我们可以发布到本地maven库,或是jcenter中心,供其他项目引用。 ### 项目结构介绍 Gradle插件开发可以用android studio,也可以用IntelliJ。AS开发需要自己创建插件项目结构,而IntelliJ可以自动生成项目结构,但是用起来一些操作不是很顺手。 如下图,用IntelliJ New出来一个项目,按照引导,即可生成我们的初始项目结构及gradle-wrapper。 ![IntelliJ](snapshot/new1.png) 这里填写插件 GroupId,ArtifactId,已经插件版本Version。如果不确定,可以先随意写个,随后可以在项目中更改。 ![IntelliJ](snapshot/new2.png) 标准的项目结构如下: ![项目结构](snapshot/xiangmu1.png) 如果用as开发,需要手动创建如上结构。从项目中可以看出支持groovy,和java的混合开发,当然从IntelliJ创建项目引导可以看出同时也是支持kotlin的。 每种语言都有各自的语言特点,比如我们开发gradle插件,在于项目build过程交互的地方用groovy开发,业务的核心代码用我们擅长的语言(java)开发,这里使用7zip的地方就是用java封装实现的。 ![项目结构](snapshot/xiangmu2.png) *resources*文件夹比较重要,这里的文件标明了插件的入口,及插件的引用名字。如果导出maven库找不到自己插件引用,可以先检查下这个文件结构是否正确。 **apk-shrink.properties** ``` implementation-class=win.canking.gradle.ShrinkPlugin ``` ``` apply plugin ‘apk-shrink’ ``` 当build.gradle 解析 apply plugin 时,就会找到 *win.canking.gradle.ShrinkPlugin*, 开始执行**apply()**方法 ### 项目可能遇到的问题 1,发布本地maven,build.gradle配置如下 ``` apply plugin: 'groovy' apply plugin: 'java' apply plugin: 'maven' group 'win.canking.gradle' version '1.0.1' archivesBaseName = 'apk-shrink'//ArtifactId默认是项目module name compileGroovy { sourceCompatibility = 1.7 targetCompatibility = 1.7 options.encoding = "UTF-8" } dependencies { compile gradleApi() compile localGroovy() compile 'com.android.tools.build:gradle:2.1.0' } uploadArchives { repositories.mavenDeployer { //文件发布到下面目录 repository(url: uri('../../ChargeHelperPro/gradleplugin/')) } } ``` 执行如下代码,可以生成本地的maven库 ``` gradlew -p {module name} clean build uploadArchives --info ``` 2,不同JDK编译问题,特别注意,需要配置sourceCompatibility 3, 引用找不到问题 先检查导出目录,是否生成了maven。目录结构如下: ![maven](snapshot/fan2.png) 反编译生成的jar包,查看打包是否正确。 ![JD-gui](snapshot/fan.png) ## APK资源自定义压缩 ### APK包结构 一个apk文件本质上就是一个zip压缩文件,我们可以用解压缩工具解压查看内部结构。 name | desp --------- |-------- res | 资源文件,该文件下资源都会映射到项目R文件中,生成引用ID assets | 静态资源文件,访问是需要用到 AssetManager lib | 第三包jar或者native so库 META-INF | 签名信息,CERT.RSA CERT.SF MANIFRST.MF AndroidManifest | 项目清单文件,包含四大组件,包信息,权限等 classes.dex | java的class文件通过dx工具生成的安卓执行文件 resources.arsc | 编译后的二进制资源文件,包含代码对资源的引用关系 ### 资源文件压缩 ``` aapt l -v xxx.apk ``` ![stored](snapshot/zipa.png) 从图中看出apk中有些资源文件存储方式为stored,是未经压缩状态,我们可以对apk再处理,通过高压缩率的工具(7zip)压缩文件,达到瘦身目的。 ### 获取7zip工具路径 通过“which 7za“获取PC上7zip工具目录 ``` ProcessBuilder pb = new ProcessBuilder(new String[]{"which", "7za"}); String sevenZipPath = ""; try { Process process = pb.start(); InputStreamReader ir = new InputStreamReader(process.getInputStream()); LineNumberReader input = new LineNumberReader(ir); String tmp; while ((tmp = input.readLine()) != null) { if (tmp.endsWith(File.separator + "7za")) { sevenZipPath = tmp; Log.a("Shrink", "7zip path:" + sevenZipPath); } } process.waitFor(); process.destroy(); } catch (Exception e) { Log.e("Error", "no shrink" + e.getMessage()); return; } ``` 定义要压缩的文件类型: ``` public class CompressInfo { public Set compressFilesPattern; private CompressInfo() { this.compressFilesPattern = new HashSet<>(); } public static CompressInfo init() { CompressInfo info = new CompressInfo(); info.compressFilesPattern.add(".png"); info.compressFilesPattern.add(".jpg"); info.compressFilesPattern.add(".JPG"); info.compressFilesPattern.add(".jpeg"); info.compressFilesPattern.add(".gif"); info.compressFilesPattern.add(".arsc"); return info; } } ``` 调用7za进行压缩目标文件 ``` private static void do7zip(String srcDirPath, String outZipPath, String sevenZipPath) throws IOException, InterruptedException { String srcFilesPath = new File(srcDirPath).getAbsolutePath() + File.separator + "*"; outZipPath = new File(outZipPath).getAbsolutePath(); ProcessBuilder pb = new ProcessBuilder(new String[]{sevenZipPath, "a", "-tzip", outZipPath, srcFilesPath, "-mx9"}); Process process = pb.start(); InputStreamReader ir = new InputStreamReader(process.getInputStream()); LineNumberReader input = new LineNumberReader(ir); String line; while ((line = input.readLine()) != null) { Log.d(input.getLineNumber() + ":" + line); } process.waitFor(); process.destroy(); } ``` 指定task来执行压缩任务 ``` @Override void apply(Project project) { Log.d("shrink apply") if (project.getPlugins().hasPlugin(AppPlugin)) { def config = project.extensions.create(SHRINK_CONFIG, ShrinkExtension) project.afterEvaluate { project.tasks.matching { println it.name it.name.startsWith('packageRelease') } each { t -> t.doLast { if (config.enable) { Log.d("shrink start...") GradleMrg.do7zipCompressApk(config.apkPath) Log.d("shrink EDN") } } } } } } ``` 也可以自定义一个单独的task,不依赖编译流程。 压缩效果对比: ![shrink](snapshot/shrinka.png)![shrink](snapshot/shrinkb.png) -------—————— 欢迎转载,请标明出处:常兴E站 [canking.win](http://www.canking.win)