Android源码探究之BaseDexClassLoader的使用

 更新时间:2022年08月10日 11:41:45   作者:失落夏天  
今天解决一个插件化问题的时候,竟然发现SO没有正常加载,很怪异,最终排查下来发现竟然是参数传入错误导致的。这就扯到了本文的标题上了,BaseDexClassLoader中的4个参数该如何传入,传入的又是什么呢

前言

一共有4个参数,分来来讲。

1:dexFile(String类型)
2:optimizedDirectory(File类型)
3:librarySearchPath(String类型)
4:parent(ClassLoader类型)

一.dexPath(String)

官方的注释:

     * @param dexPath the list of jar/apk files containing classes and
     * resources, delimited by {@code File.pathSeparator}, which
     * defaults to {@code ":"} on Android.

包含类和类的jar/apk文件列表资源,由{@code File.pathSeparator}分隔,其中Android上的默认值为{@code”:“}。

也就是说这里其实是可以传入一个集合的。

比如如下的参数都是可以被接受的:

sdcard/test/aa.apk

sdcard/test/aa.apk:sdcard/test/bb.apk:sdcard/test/cc.apk

sdcard/test/aa.apk:sdcard/test/dd.jar

其中分隔符:,保险起见,可以使用File.pathSeparator替代。

示例代码如下:

private void loadDex(List<File> apkList) {
        StringBuilder builder = new StringBuilder();
        for (File f : apkList) {
            builder.append(f.getAbsolutePath());
            builder.append(File.separatorChar);
        }
        DexClassLoader dexClassLoader = new DexClassLoader(builder.toString(), null, null, getClass().getClassLoader());
    }

二.optimizedDirectory

官方的注释:

this parameter is deprecated and has no effect since API level 26.

解压的路径,这里传入路径的最主要目的就是为了生成odex文件夹,方便后续存储odex文件。

如注释中所写,这个参数26开始已经失效了。所以这里就不扩展去讲了。

三.librarySearchPath

官方的注释:

 * @param librarySearchPath the list of directories containing native

包含native目录的目录列表,这里要注意的,传入的一定是so的上一级目录才可以。如果是更上一层的目录是不行的。前言中的问题就出在这,传入了一个更上一层的目录地址。

排查流程:

最终使用librarySearchPath的地方是在DexPathList的splitPaths方法中。生成File加入List中:

@UnsupportedAppUsage
    private static List<File> splitPaths(String searchPath, boolean directoriesOnly) {
        List<File> result = new ArrayList<>();
        if (searchPath != null) {
            for (String path : searchPath.split(File.pathSeparator)) {
                ...
                result.add(new File(path));
            }
        }
        return result;
    }

而使用的时候,是使用了findLibrary的方法,for循环便利上面集合中的所有path,看是否存在。

public String findLibrary(String libraryName) {
        String fileName = System.mapLibraryName(libraryName);
        for (NativeLibraryElement element : nativeLibraryPathElements) {
            String path = element.findNativeLibrary(fileName);
            if (path != null) {
                return path;
            }
        }
        return null;
    }

而寻找的最终会调用到NativeLibraryElement的findNativeLibrary方法:

public String findNativeLibrary(String name) {
            maybeInit();
            if (zipDir == null) {
                String entryPath = new File(path, name).getPath();
                if (IoUtils.canOpenReadOnly(entryPath)) {
                    return entryPath;
                }
            } else if (urlHandler != null) {
                // Having a urlHandler means the element has a zip file.
                // In this case Android supports loading the library iff
                // it is stored in the zip uncompressed.
                String entryName = zipDir + '/' + name;
                if (urlHandler.isEntryStored(entryName)) {
                  return path.getPath() + zipSeparator + entryName;
                }
            }
            return null;
        }

这段代码也就是问题的核心了,直接使用了new File(path,name)的方式,而不是循环查找,所以只有上一层才可以。

四.parent

官方的注释:

@param parent the parent class loader

这个比较简单,就是上一层的classLoader。

五.总结

无论是dexPath还是librarySearchPath,都是支持多路径传入的。路径之间使用File.pathSeparator进行分隔。dexPath中必须是APK或者Jar包的路径,而librarySearchPath中必须是so文件的上一层级文件夹才可以。

到此这篇关于Android源码探究之BaseDexClassLoader的使用的文章就介绍到这了,更多相关Android BaseDexClassLoader内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • 使用RecyclerView实现水平列表

    使用RecyclerView实现水平列表

    这篇文章主要为大家详细介绍了使用RecyclerView实现水平列表,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2019-09-09
  • 在android中如何用Java加载解析so

    在android中如何用Java加载解析so

    我们在android开发项目过程中都必然会更so加载打交道,那么so加载在系统中的顺序和流程是怎样的,我们就有必要对这个加载过程进行熟悉了解掌握
    2021-10-10
  • Android发生ANR后的信息采集过程

    Android发生ANR后的信息采集过程

    这篇文章主要为大家介绍了Android发生ANR后的信息采集过程,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2023-07-07
  • Android实现显示和隐藏密码功能的示例代码

    Android实现显示和隐藏密码功能的示例代码

    在前端中我们知道用javascript就可以可以很容易实现密码的显示与隐藏,本文将大家详细介绍Android是如何实现显示和隐藏密码功能的,需要的可以参考一下
    2022-06-06
  • 图文详解Android Studio搭建Android集成开发环境的过程

    图文详解Android Studio搭建Android集成开发环境的过程

    这篇文章主要以图文的方式详细介绍了Android Studio搭建Android集成开发环境的过程,文中安装步骤介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2015-12-12
  • Android自定义View实现抽奖转盘

    Android自定义View实现抽奖转盘

    这篇文章主要为大家详细介绍了Android自定义View实现抽奖转盘,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2018-12-12
  • Android自定义View实现等级滑动条的实例

    Android自定义View实现等级滑动条的实例

    这篇文章主要介绍了 Android自定义View实现等级滑动条的实例的相关资料,需要的朋友可以参考下
    2017-04-04
  • Android基础知识之broadcast广播详解

    Android基础知识之broadcast广播详解

    这篇文章主要为大家详细介绍了Android基础知识之broadcast广播的相关资料,感兴趣的小伙伴们可以参考一下
    2016-06-06
  • Android 键盘开发知识点总结

    Android 键盘开发知识点总结

    这篇文章我们给大家总结了Android 键盘开发的相关知识点内容以及开发心得,有需要的朋友参考学习下。
    2018-06-06
  • Android实现图片浮动随意拖拽效果

    Android实现图片浮动随意拖拽效果

    这篇文章主要介绍了Android的图片在界面随意拖动的功能,小编觉得挺不错的,现在分享给大家,也给大家做个参考,一起跟随小编过来看看吧
    2018-04-04

最新评论