分析so文件和依赖的关系

53 篇文章 1 订阅
46 篇文章 1 订阅

1、前言

在包大小的占比中,so文件的占比往往是最高的,动辄几兆的大小多一个都会把包大小的指标打爆。

而在各厂商要求对手机CPU ARM架构进行分包适配的情况下,你更需要知道哪些依赖是没有适配v7a/v8a的,这将影响你的APP在应用市场的审核。

所以搞清楚so文件和依赖的关系,它不仅是一个技术指标归因的工具,也是应对厂商分包适配的利器。

2、分析APK

我们一般分析APK是通过Android Studio提供的Analyze APK工具,可以清晰的看到APK文件的组成部分,比如lib文件夹下有哪些so文件,但是却无法直观的看出这些so文件属于哪个依赖。

如下图:

3、so文件怎么来的

想要知道so文件是属于哪个依赖,那么得先搞清楚so文件是怎么来的。

如果你分析过apk里面的so文件,你会发现,除了项目中lib文件夹下手动添加的so文件之外,还有一些不知道是哪来的。

要想搞清楚这个,你还得知道我们的依赖是依赖的什么东西。

以okhttp为例:

implementation 'com.squareup.okhttp3:okhttp:4.10.0'

我们通过GAV坐标依赖的实际是square公司发布的jar/aar文件。

图片

ok,整体思路我们捋一下:

Gradle管理依赖会自动去下载jar/aar,然后我们通过遍历所有依赖,拿到对应的jar/arr,再去获取其中对应的so文件。

听起来并不复杂,实际上也很简单,下面来实战一下。

4、实战

4.1、栗子

以阿里云音视频SDK为例,它一定是包含so文件的,我不信它不用FFmpeg🐶。

添加示例依赖:

    //8.全功能:直播推流(含超低延时直播、RTC连麦)+短视频+播放器+美颜特效
    implementation 'com.aliyun.aio:AliVCSDK_Premium:6.4.0'

其他配置就不赘述了,可以自行去看文档。

别忘了添加ndk配置,否则so打不进去。

defaultConfig {
    ndk {
        abiFilters 'arm64-v8a', 'armeabi-v7a', 'armeabi-v8a'
    }
}

4.2、遍历

ok,准备工作到位,写个插件遍历所有的依赖文件并打印出来。

Configuration configuration = project.getConfigurations().getByName(applicationVariant.getName() + "CompileClasspath");
configuration.forEach(file -> {
    System.out.println(TAG + "file " + file.getName());
    String fineName = file.getName();
    if (fineName.endsWith(".jar") || fineName.endsWith(".aar")) {
        try {
            JarFile jarFile = new JarFile(file);
            for (Enumeration enums = jarFile.entries(); enums.hasMoreElements(); ) {
                JarEntry jarEntry = (JarEntry) enums.nextElement();
                System.out.println(TAG + "jarEntry " + jarEntry.getName());
            }
        } catch (IOException e) {
            throw new RuntimeException(e);
        }
    }
});

这里在Project评估完成之后,获取Configuration对象,然后遍历所有的依赖文件,最后通过JarEntry把文件里面所有的子文件打出来。

输出:

图片

可以看到,已经把所有的子文件都打出来了,包括jni/、res/、assets/等等。

4.3、优化

上面的输出还不够直观,我们在过滤一下,只打印so文件,然后优化一下打印的格式。

configuration.forEach(file -> {
    String fineName = file.getName();
    System.out.println(TAG + "fine name = " + fineName);
    if (fineName.endsWith(".jar") || fineName.endsWith(".aar")) {
        try {
            JarFile jarFile = new JarFile(file);
            for (Enumeration enums = jarFile.entries(); enums.hasMoreElements(); ) {
                JarEntry jarEntry = (JarEntry) enums.nextElement();
                if (jarEntry.getName().endsWith(".so")){
                    System.out.println(TAG + "----- so name = " + jarEntry.getName());
                }
            }
        } catch (IOException e) {
            throw new RuntimeException(e);
        }
    }
});

最终效果:

//...
GradleXPlugin >>>>> fine name = AliVCSDK_Premium-6.4.0.aar
GradleXPlugin >>>>> ----- so name = jni/arm64-v8a/libMNN_CL.so
GradleXPlugin >>>>> ----- so name = jni/arm64-v8a/libalivcffmpeg.so
GradleXPlugin >>>>> ----- so name = jni/arm64-v8a/liball_in_one.so
GradleXPlugin >>>>> ----- so name = jni/arm64-v8a/libMNN.so
GradleXPlugin >>>>> ----- so name = jni/armeabi-v7a/libMNN_CL.so
GradleXPlugin >>>>> ----- so name = jni/armeabi-v7a/libalivcffmpeg.so
GradleXPlugin >>>>> ----- so name = jni/armeabi-v7a/liball_in_one.so
GradleXPlugin >>>>> ----- so name = jni/armeabi-v7a/libMNN.so
//...

是不是还挺简单的~

5、最后

如果你不想自己写,这个插件我也发布远端了,按照下面三步走,即可使用。

Step 1. Add the JitPack repository to your build file

repositories {
    ...
    maven { url 'https://jitpack.io' }
}

Step 2. Add the dependency

dependencies {
    classpath('com.github.yechaoa.GradleX:plugin:1.2')
}

Step 3. Add the Plugin Id to your build file and configure the gradleX{ } dsl

plugins {
    id 'com.yechaoa.plugin.gradleX'
}

gradleX {
    printDependencies = false
    analysisSo = true
}

6、GitHub

github.com/yechaoa/Gra…

作者:yechaoa
链接:https://juejin.cn/post/7287429638019448888
来源:稀土掘金
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 解析.so是一个类似于depends工具的工具,用于解析和分析在Linux系统中使用的动态链接库文件(.so文件)的依赖关系。 在Linux系统中,动态链接库文件是由程序在运行时需要的共享库文件,用于提供程序所需的功能和资源支持。解析.so工具可以将.so文件作为输入,分析其中的符号引用和依赖关系。 首先,解析.so工具可以识别.so文件中的符号引用,这些符号引用是程序运行时需要调用的函数或变量。工具可以将这些符号引用与系统库或其他共享库中的符号关联起来,以确保程序可以正确地调用所需的函数和变量。 其次,解析.so工具还可以分析.so文件依赖关系。动态链接库文件往往会依赖于其他的库文件,这些文件提供了.so文件所需要的函数和资源。解析.so工具可以检测并列出这些依赖关系,以帮助开发人员了解和管理程序所需的库文件。 通过解析.so工具,开发人员可以更好地理解程序所需的动态链接库文件的相关信息。他们可以识别程序中使用的库文件和函数,了解库文件之间的依赖关系,并及时处理可能出现的依赖问题。这有助于确保程序能够正确地加载和运行,并提供所需的功能支持。 因此,解析.so是一个类似于depends工具的工具,它可以帮助开发人员分析和管理程序运行时的动态链接库文件的符号引用和依赖关系,从而提高程序的稳定性和可靠性。 ### 回答2: 解析.so是一个类似于depends工具的工具。它可以用来分析查看一个.so文件依赖关系。 在计算机编程中,动态链接库(Dynamic Link Library,简称DLL)是一种可以在运行时被程序动态加载和链接的文件。而.so文件则是在类Unix系统(比如Linux)中使用的共享对象文件格式,也是一种动态链接库。 解析.so工具可以将一个.so文件打开,并显示出该文件中的所有依赖项。这些依赖项通常是其他.so文件或系统库文件,它们被称为该.so文件依赖关系。 类似于depends工具,解析.so可以帮助开发人员和系统管理员了解一个程序运行所需的各种依赖项,以便在部署和运维过程中更好地管理和维护软件。同时,解析.so还可以帮助开发人员排除一些依赖问题,比如找到缺少的依赖项或者解决依赖冲突等。 总的来说,解析.so是一个方便的工具,可以帮助开发人员和系统管理员更好地理解和管理.so文件依赖关系。它在软件开发、部署和维护过程中都有很大的作用,提供更好的可靠性和可维护性。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值