浅谈Android Classloader动态加载分析
作者:JohnsonZZZ 发布时间:2022-01-28 13:18:05
ClassLoader概念
我们知道,Java源文件(.java)经过编译器编译之后,会转换成Java字节码(.class),然而程序是如何加载这些字节码文件到内存中呢?这就用到了ClassLoader,即类加载器。ClassLoader类加载器负责读取 Java 字节代码,并转换成 java.lang.Class类的一个实例。从而只有class文件被载入到了内存之后,才能被其程序所引用。所以ClassLoader就是用来动态加载class文件到内存当中用的。
ClassLoader的分类
Android中的常用几种类加载器类型继承关系划分可以用一组关系图来表示
BootClassLoder
通过查看ClassLoader源码 我们得知,Android中在默认父加载器传入的情况下,默认父加载器为PathClassLoder,而PathClassLoader的父加载器正是BootClassLoader。BootClassLoader是ClassLoader的内部类,是包内可见,我们无法直接使用,也无法直接动态加载。
/**
* Encapsulates the set of parallel capable loader types.
*/
private static ClassLoader createSystemClassLoader() {
String classPath = System.getProperty("java.class.path", ".");
String librarySearchPath = System.getProperty("java.library.path", "");
...省略部分代码
//默认父构造器为PathClassLoder
return new PathClassLoader(classPath, librarySearchPath, BootClassLoader.getInstance());
}
URLClassLoader
URLClassLoader继承自SecureClassLoader,SecureClassLoader继承自ClassLoader。URLClassLoader的特点就是只能加载jar文件,但是dalvik不能直接识别jar。所以在Android中无法直接使用这个类加载器。
BaseDexClassLoader
BaseDexClassLoader直接继承自ClassLoader,下面是其构造函数
public BaseDexClassLoader(String dexPath, File optimizedDirectory, String librarySearchPath, ClassLoader parent) {
super(parent);
this.pathList = new DexPathList(this, dexPath, librarySearchPath, null);
....
}
下面解析下这四个参数
dexPath:指目标类所在的apk、dex或者jar文件的路径(包括SD卡),然后加载器将从该路径中寻找到指定的目标类。当然了,这个路径可以是多个路径,这样可以寻找到多个目标类,多路径之间需要使用特定的分隔符,分隔符可以使用System.getProperty("path.separtor")获取。
optimizedDirectory:由于dex文件被包含在apk或者jar文件中,需要先解压出来,而这个参数 就代表了被解压的路径。而且apk文件其实也是一个压缩包,解压的过程其实也是一个ODEX优化的过程,那么何为ODEX优化呢?其实就是把包里面的可执行程序提取出来变成ODEX文件,存放到optimizedDirectory目录下,因为提取出来的原因,应用第一次进行启动的时候,直接使用ODEX文件 启动速度自然是比解压再启动速度是要快的。为什么是应用第一次启动呢?因为dex版本只有第一次启动会解压执行程序到/data/dalvik-cache(针对PathClassLoader),或者optimizedDirectory文件目录下(针对DexClassLoader),之后就可以直接读取目录下的dex文件了。
librarySearchPath:指的是目标类所使用的c、c++库存放的路径
parent:是指该加载器的父加载器,一般为当前执行类的加载器。
PathClassLoader
public PathClassLoader(String dexPath, ClassLoader parent) {
super(dexPath, null, null, parent);
}
public PathClassLoader(String dexPath, String librarySearchPath, ClassLoader parent) {
super(dexPath, null, librarySearchPath, parent);
...
}
通过源码我们可以知道,PathClassLoader继承于BaseDexClassLoader,并且构造器将optimizedDirectory置为null,也就是没有设置ODEX优化后的存储路径,前文有提到,如果没有设置optimizedDirectory目录,那么默认存储路径就是/data/dalvik-cache。因为这个原因,PathClassLoader被设定成只能加载Android系统类和已安装的android应用类。
DexClassLoader
public DexClassLoader(String dexPath, String optimizedDirectory, String librarySearchPath, ClassLoader parent) {
super(dexPath, new File(optimizedDirectory), librarySearchPath, parent);
...
}
DexClassLoader也是继承于BaseDexClassLoader,支持加载包含classes.dex文件的apk、jar,前文我们提到dalvik不支持直接加载jar文件,那么为什么到了DexClassLoader这里怎么就可以支持加载了呢?原因在于其父类BaseDexClassLoader对于“.jar”,“.apk”,".zip",".dex"后缀的文件都会进行对应的处理,最终提取成可执行的dex文件。然而URLClassLoader并未对此做类似的处理,因此我们一般会采用DexClassLoader做动态加载。
InMemoryDexClassLoader
public InMemoryDexClassLoader(ByteBuffer[] dexBuffers, ClassLoader parent) {
super(dexBuffers, parent);
}
public InMemoryDexClassLoader(ByteBuffer dexBuffer, ClassLoader parent) {
this(new ByteBuffer[] { dexBuffer }, parent);
}
InMemoryDexClassLoader继承于BaseDexClassLoader,是API26新增的类加载器。dexBuffers数组构造了一个DexPathList,可用于加载内存中的dex。
DelegateLastClassLoader
public DelegateLastClassLoader(String dexPath, ClassLoader parent) {
super(dexPath, parent);
}
DelegateLastClassLoader是API27新增的类加载器,继承自 PathClassLoader。DelegateLastClassLoader实行最后的查找策略。使用DelegateLastClassLoader来加载每个类和资源,使用的是以下顺序:
判断是否已经加载过该类
搜索此类的类加载器是否已经加载过该类
搜索与此类加载器相关联的dexPath文件列表,并委托给父加载器。
双亲委托机制
Android类加载器通过loadClass加载目标类,下面是加载的源码
protected Class<?> loadClass(String name, boolean resolve)
throws ClassNotFoundException
{
// 首先检查当前目标类是否已经被加载过,有则直接返回
Class<?> c = findLoadedClass(name);
if (c == null) {
try {
if (parent != null) {
//如果有父类加载器,优先使用父类加载器寻找目标类
c = parent.loadClass(name, false);
} else {
//其次,如果有辅助类加载器,使用辅助类加载器寻找目标类
c = findBootstrapClassOrNull(name);
}
} catch (ClassNotFoundException e) {
// ClassNotFoundException thrown if class not found
// from the non-null parent class loader
}
if (c == null) {
//如果仍未找到,则通过寻找子ClassLoader的目标类(如果子ClassLoader重写了findClass)
c = findClass(name);
}
}
return c;
}
由上述源码,我们可以总结:
当前类加载器首先检查目标类是否已经被加载过,有则直接返回
当前类加载器会先委托父类加载器加载目标类,如果未设置父加载器,则检查辅助加载器是否支持查询加载目标类
只有上述加载器找不到目标类的时候,才会调用当前类加载器(Child) 查询路径寻找目标类。
以上这么做的好处是:一方面防止目标类的重复加载,另外一方面 主要考虑安全因素,防止有人重写原生类,比如说java.lang.String这样的数据类型,替换原生的String类,加载到JVM中,造成严重的安全问题。
双亲委托机制 在Android热修复领域中也有着广泛的应用。每个ClassLoader可以有多个dex文件,每个dex文件是一个Element,多个dex文件组成一个dexElements,类加载器寻找类的时候,会遍历dexElements中的dex文件,再通过dex文件遍历目标类。由于双亲委托机制的存在,寻找到目标类后就直接返回,不再寻找其他dex文件下该目标类,热修复的原理就是hook住ClassLoader,使其先加载修复后的目标类,而存在的BUG的目标类不会被加载。
来源:https://www.jianshu.com/p/f6aa92cdf4f4


猜你喜欢
- 本文实例为大家分享了Android五子棋游戏的具体代码,供大家参考,具体内容如下1、效果图:2、GobangPanel棋盘面板:public
- 本文浅析了C#的复制和克隆技术,对于有需要的朋友可以参考下。在C#中,用HashTable,DataTable等实现复制和克隆,下面直接看例
- 问题描述:在用fabric集成后编译出现如下错误,Error:Cause: hostname in certificate didn'
- System.getProperty()的作用及使用最近在看一些代码时,很多地方都用到了System.getProperty()、Syste
- 模板消息文档公众号的类型分为服务号、订阅号和企业号,其中服务号和订阅号比较常见。要想实现公众号推动消息给指定的用户,其类型必须为服务号。推送
- 目前市面上的第三方短信服务平台数量十分庞大,并能为企业提供三网合一的短信接口,这些服务商通过整合短信发送服务,简单的打包成为API和SDK接
- 线程可以理解为下载的通道,一个线程就是一个文件的下载通道,多线程也就是同时开启好几个下载通道。当服务器提供下载服务时,使用下载者是共享带宽的
- 目的官方的Drools范例大都是基于纯Java项目或Maven项目,而基于Spring Boot项目的很少。本文介绍如何在Spring Bo
- 在实际项目开发中,对于Excel的导入导出还是很常见的需求,比如说将数据根据模板批量导入到数据库中,以及将数据库中的数据批量导出陈Excel
- java @Value("${}")获取不到配置文件中值1、property.yml配置spring: ma
- fifter、servlet、interceptorfifter用来处理请求头、请求参数、编码的一些设置,然后转交给servlet,处理业务
- 一、问题反馈今天公司测试向我反馈,系统用户模糊查询功能在用户名称包含特殊字符时(_、\、%)无法正常查询结果。二、问题验证1、当like中包
- 首先说一下,教科书上的扫描线算法确实是用c++很好实现,而且网上有很多源码,而java实现的基本没有(可能是我没看到),所以肖先生还是打算自
- 本文实例为大家分享了java实现打印正三角的具体代码,供大家参考,具体内容如下代码:package BasicType;/** * 封装一个
- 基本介绍BottomSheetDialog是底部操作控件,可在屏幕底部创建一个支持滑动关闭视图。目前依赖使用如下:implementatio
- Spring中Profile对不同环境提供不同配置功能的支持,可以通过激活、指定参数等方式快速切换环境。文件名格式:application-
- 前言:在日常开发中经常会遇到字符串匹配问题,我们就来学习使用Java中的一些方便快捷的方法来解决这个问题吧使用String类Java自带的字
- 一、AXIS调用远程WebService,以国内手机号归属地查询为例 1、wsdl地址:http://ws.webxml.com.
- DSA数字签名,供大家参考,具体内容如下一、实验目的在掌握了ElGamal和Schorr数字签名算法的基础上,进一步地学习和掌握DSA签名算
- 本文以实例形式讲述了C#泛型的用法,有助于读者深入理解C#泛型的原理,具体分析如下:首先需要明白什么时候使用泛型:当针对不同的数据类型,采用