Android-So-Handler
                                
                                 Android-So-Handler copied to clipboard
                                
                                    Android-So-Handler copied to clipboard
                            
                            
                            
                        支持APK包中所有System.Load/LoadLibrary加载的So库文件(Maven、aar文件引入三方库、源码引入的so文件),对So文件进行7z压缩与云端下发,完成apk瘦身
Android-So-Handler
Jitpack 接入方式
因为接入 Jitpack 后,暂时没有合并到主仓库,所以接入方式(maven 仓库链接)需要先按下面方法修改,其他的依旧按原仓库说明操作。
buildscript {
    repositories {
        maven("https://jitpack.io")
    }
    dependencies {
        // ...
        if (userSoPlugin) {
            classpath("com.github.mainlxl.Android-So-Handler:load-hook-plugin:${Versions.so_plugin_version}")
            classpath("com.github.mainlxl.Android-So-Handler:file-plugin:${Versions.so_plugin_version}")
        }
    }
}
repositories {
    maven("https://jitpack.io")
}
dependencies {
    implementation "com.github.mainlxl.Android-So-Handler:load-hook:${Versions.so_plugin_version}"
    implementation "com.github.mainlxl.Android-So-Handler:load-assets-7z:${Versions.so_plugin_version}"
}
** 减包工具集合 , 通过处理 so 库实现减包 **
PS:减 so 文件时很有必要了解 so 来自那个三方或者一方库知己知彼,这里推荐我另一个项目 AnalyzeSoPlugin 去溯源
特点如下:
- 支持 APK 中所有通过 System.Load/LoadLibrary加载的 So 库文件(包含 Maven、aar 等方式引入三方库与源码中的 so 文件)进行处理。
- 支持 7z 压缩与云端下发
- 对项目代码侵入少,如果只是压缩 so
库,只需一行初始化 AssetsSoLoadBy7zFileManager.init(v.getContext());即可。
- 云端下发 so 库需要在 init中传入NeedDownloadSoistener自行下载,并在下载后调用SoFileInfo#insertOrUpdateCache(saveLibsDir,File)插入缓存即可,** 需要在加载前插入缓存 **
数据对比:
仅仅在加载时通过记录的 MD5 判断是否存在,不存在解压,存在则跳过直接加载。 这里通过压缩时记录的 MD5 判断是否需要解压更新不依赖 apk 版本减少解压次数。
| so 库名称 | apk 包中所占大小 | 7z 极限压缩大小 | 解压后实际大小 | 解压耗时 (毫秒) | 
|---|---|---|---|---|
| RTCStatistics | 1331kb(1.3M) | 958kb | 2752kb(2.68M) | 109 | 
| flutter(Debug 版本) | 10,547kb(10.3M) | 6360kb | 23358kb(22.8M) | 700 | 
| bd_idl_pass_token | 9.6k | 8kb | 17kb | 3 | 
| idl_license | 63.3kb | 51kb | 113kb | 6 | 
| FaceSDK | 269.5kb | 220kb | 450kb | 25 | 
| turbonet | 1,638.4kb(1.6M) | 1258kb | 2737kb | 167 | 
| gnustl_shared | 273.7kb | 195kb | 693kb | 28 | 
| xiaoyuhost | 426.9kb | 309kb | 1009kb(1M) | 48 | 
| crab_native | 57.7kb | 44kb | 109kb | 7 | 
apk 包中所占大小: apk 属于 zip 压缩 所以 apk 包中已经为 zip 压缩后大小
7z 极限压缩大小: 7z 极限压缩大小是执行 7z a xxx.7z libxxx.so -t7z -mx=9 -m0=LZMA2 -ms=10m -mf=on -mhc=on -mmt=on -mhcf 压缩后大小
** 解压后实际大小:** 指 so 文件实际大小,AndroidStudio 中文件大小
** 解压耗时 (毫秒):** 统计手机为谷歌 Pixel 2XL 骁龙 835 处理器
接入方式如下:
ps: 配置较多全可走默认 ~_ ~!
- 
前往 Release 下载对应版本 maven.zip解压并放入项目根目录
- 
根 build.gradle 中加入 
buildscript {
    repositories {
        maven { url uri("${rootDir}/maven") }
    }
}
allprojects {
    repositories {
        maven { url uri("${rootDir}/maven") }
    }
}
- 
复制工程下 so-file-config.gradle 到工程根目录 
- 
工程根目录 gradle.properties 中添加 SO_PLUGIN_VERSION=x.x.xbuild.gradle 中添加classpath "com.imf.so:load-hook-plugin:${SO_PLUGIN_VERSION}"和classpath "com.imf.so:file-plugin:${SO_PLUGIN_VERSION}"x.x.x 修改为从 Release 下载的版本 如:0.0.7 
- 
app 的 build.gradle 中添加 apply from: "${rootDir}/so-file-config.gradle"
- 
在 Application 中调用 AssetsSoLoadBy7zFileManager.init(v.getContext());初始化, 重载方法支持传入 NeedDownloadSoListener 完成云端所需要 so 库下载, 下载后使用 SoFileInfo#insertOrUpdateCache( saveLibsDir,File) 插入缓存中
- 
修改根目录中 so-file-config.gradle 进行压缩删减库配置主要修改 deleteSoLibs 与 compressSo2AssetsLibs 如下: 
// 指定编辑阶段要删除的 so 库
deleteSoLibs = []
// 指定至 assets 中的 so 库
compressSo2AssetsLibs = []
** 其他配置请参考注释 **
插件介绍
一、 SoLoadHookPlugin 插件
- 
通过 Hook System.loadLibrary与System.load实现加载转发具体步骤如下:- 通过 ASM框架对 Hook 目标类进行字节码修改具体为System.loadLibrary与System.load修改成SoLoadHook.loadLibrary与SoLoadHook.load
- SoLoadHook可设置- SoLoadProxy完成对外代理
 SoLoadHook有默认实现只是调用System.loadLibrary与System.load
- 通过 
- 
具体接入步骤如下: 
- gradle 配置
//build.gradle 中只加入
classpath "com.imf.so:load-hook-plugin:${SO_PLUGIN_VERSION}"
//app.gradle 中只配置
apply plugin: 'SoLoadHookPlugin'
SoLoadHookConfig {
    // 是否跳过 R 文件与 BuildConfig
    isSkipRAndBuildConfig = true
    // 设置跳过的包名, 跳过的包不去 hook 修改后请先 clean
    excludePackage = ['com.imf.test.']
}
dependencies {
    implementation "com.imf.so:load-hook:${SO_PLUGIN_VERSION}"
}
- java 代码实现 SoLoadProxy完成加载
public interface SoLoadProxy {
    void loadLibrary(String libName);
    void load(String filename);
}
SoLoadHook.setSoLoadProxy(new XXXSoLoadProxy())
实现 SoLoadProxy 类后不会被修改
System.loadLibrary与System.load字节码 如果不想在指定包名下修改 在 excludePackage 中配置报名 如果不想在指定类或方法下被修改字节码, 请添加注解 @KeepSystemLoadLib
二、~~SoFileTransformPlugin 与 SoFileAttachMergeTaskPlugin 插件依赖
SoLoadHookPlugin~~, 使用 ApkSoFileAdjustPlugin
~~SoFileTransformPlugin~~ 与 ~~SoFileAttachMergeTaskPlugin~~ 功能一样只是编辑阶段插入口不同 根据 com.android.tools.build:gradle:x.x.x 中版本号不同选择使用哪个 3.4.0 版本及以下使用 SoFileTransformPlugin 3.5.0 - 3.6.0 版本使用 SoFileAttachMergeTaskPlugin 4.1.0 以上包含 7.3.0 版本使用 ApkSoFileAdjustPlugin
4.1.0 中添加了 compressed_assets 机制导致无法把压缩后的 so 文件放入 asstes 中顾调整为针对已出包 apk 进行 so 文件操作并重新签名
- 通过实现 transform 或在 mergeNativeLibs 中添加 Action 的方式, 对 so 库进行 7z 压缩 (利用压缩差实现压缩
apk), 压缩后放入 asstes下的jniLib
- 根据压缩或删除 so 情况生成 info.json
- 运行时进行解压加载 so
- so 库依赖拷贝了 ReLinker 中解析代码
- 解压部分微调自 AndroidUn7z
** 接入方式参考顶部最开始部分 **
三、常见问题
- 
安装时报错 Failure [INSTALL_FAILED_INVALID_APK: Failed to extract native libraries, res=-2]请在 application标签添加属性android:extractNativeLibs="true"如下:<?xml version="1.0" encoding="utf-8"?> <manifest ...> <application android:extractNativeLibs="true"> ... </manifest>关于 extractNativeLibs 属性 - 进行 apk 打包时,android:extractNativeLibs=false会对 Module 中的 so 库进行压缩,最终得到的 apk 体积较小。 >
 - 好处是:用户在应用市场下载和升级时,因为消耗的流量较小,用户有更强的下载和升级意愿。
 - 缺点是:因为 so 是压缩存储的,因此用户安装时,系统会将 so 解压出来,重新存储一份。因此安装时间会变长,占用的用户磁盘存储空间反而会增大。
- minSdkVersion < 23 或 Android Gradle plugin < 3.6.0,打包时默认值- android:extractNativeLibs=true;
- minSdkVersion >= 23 并且 Android Gradle plugin >= 3.6.0,打包时默认值- android:extractNativeLibs=false;
 apk 对比 7z 是本插件压缩后版本 false/true 代表 extractNativeLibs 属性版本,可以下载 apk 拖入 Android Studio 查看如:  apk 路径 apk 大小 下载大小 app-debug-7z.apk 2.5MB 2.4MB app-debug-false.apk 3.8MB 2.6MB app-debug-true.apk 3.1MB 2.6MB 
- 进行 apk 打包时,
感谢mcxinyu的贡献。也欢迎加我微信进行交流 x
