您的位置:首页 > 编程语言 > Java开发

理解Java虚拟机(3)之.class文件加载双亲委派模型

2016-10-18 19:06 477 查看

理解Java虚拟机(3)之.class文件加载双亲委派模型

读《深入理解Java虚拟机》-周志明 读书笔记



双亲委派模型是.class文件加载过程中一种加载机制,classLoader大致可以分成这4个层级,加载的过程中,首先将某个类一层一层传给上一层加载器,直到顶层,顶层没找到这个类,顶层找到了这个类就直接加载,没有的话会一层一层传下来,每层分别寻找,找到了,就加载,没有就接着往下传,详细说明下,每一层的作用如图

Bootstrap ClassLoader(启动类加载器)

Bootstrap Classloader负责加载/lib目录下的,并且能被虚拟机识别的(自定义一个aaaaaa.jar,Bootstrap是不会加载到,想想也是,要是随便放个jar在里面就加载了,那虚拟机肯定会出各种问题)的类库加载到虚拟机内存,Bootsstrap无法被java程序直接引用,用户如果将自己的类要委派给其加载,值接将类加载赋值为null.,这个加载器是用C++写的

Extension ClassLoader(扩展类加载器)

负责加载/lib/ext目录下中jar包

Application ClassLoader(应用类加载器)

我们安装Jdk时配置在classpath中的就是这个加载器,也叫系统加载器,负责加载classpath下文件,常用的ClassLoader.getSystemClassLoader(),就是这个加载器

##自定义加载器

自己定义一个类继承ClassLoader,重写loadClass()方法,就可以了,有自己这个loadClass方法加载自己的类,像junit,容器toncat,jeetty都是是有自己的ClassLoader,截取部分junit的ClassLoader如下:

public class TestCaseClassLoader
extends ClassLoader

public synchronized Class loadClass(String name, boolean resolve)
throws ClassNotFoundException
{
Class c = findLoadedClass(name);
if (c != null) {
return c;
}
if (isExcluded(name)) {
try
{
return findSystemClass(name);
}
catch (ClassNotFoundException localClassNotFoundException) {}
}
if (c == null)
{
byte[] data = lookupClassData(name);
if (data == null) {
throw new ClassNotFoundException();
}
c = defineClass(name, data, 0, data.length);
}


四种加载器的加载顺序,原理

如果有自定义加载器情况下,自定义类加载器收到加载请求,它首先不会自己尝试加载这个类,而是把这个请求委派给父类,也就是Application ClassLoader,Application ClassLoader会传给Extension ClassLoader一直到顶层的Bootstrap ClassLoader,如果Bootstrap ClassLoader发现有这个类就会加载(例如java.lang.Object),如果没有比方自定义的(com.hc.test),就会往下传,Extension要是发现自己有就加载,没有的话,接着往下传,知道自定义类加载器加载,如果没有自定义加载器,就是从Application ClassLoader开始,这就是双亲委派模型原理.

双亲委派的设计由衷可能是这样的,避免了加载混乱,比方自己定义了一个java.lang.object,JVM也有这个类,如果没有这个验证机制,把自己写的obejct也加载了,那么程序就无法识别是哪个object类了,系统也就混乱了.
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息