抽取壳
ClassLoader
ClassLoader用来加载Java类到 Java 虚拟机中,能够加载 jar/apk/dex。ClassLoader继承自BaseDexClassLoader。
BaseDexClassLoader的构造函数中创建了一个DexPathList类,该类负责加载文件的类。
在 DexPathList 的构造方法中,有一个方法 makeDexElements,makeDexElements 方法判断文件后缀名是否是“dex”,如果是就调用 LoadDexFile 方法加载文件,并且返回一个 DexFile 对象。
而我们的 Dex 文件就是在生成这个 DexFile 对象调用它的构造方法时候,被更具体和底层地进行了加载。在 DexFile 的构造方法中调用了一个 native 方法 openDexFile。这个 native 方法返回了一个极为重要的值 cookie。
cookie 在 Java 层就是虚拟机的 cookie 值,在 so 层它是 pDexOrJar 指针,虚拟机在进行查找 Dex 文件中的类方法时候,都是需要对 cookie 进行操作的。
查找的过程是,BaseDexClassLoader类 的 findClass 方法,调用 DexPathList类 的 findClass 方法,然后调用刚才说到的返回的 DexFile 对象的 loadClassBinaryName 方法。在 loadClassBinaryName 方法中把查找的类方法名称、类加载器、cookie 值,作为三个参数传进了最后的 defineClass 方法,这个方法调用了 native 函数来返回一系列调用想要查找的 class。然后就是 loadClass 和 defineClass 的操作。
BaseDexClassLoader类findClass方法:
DexPathList类findClass方法:
DexFile类loadClassBinaryName方法: