在前面两篇博客中我们简单介绍了类加载器的基础和类的生命周期的基础内容,今天我们来继续深入的来看一下java的类加载器的详细内容。我们都知道。类加载器用来把类加载到 java 虚拟机。从 JDK2.0 开始,类的加载过程采用父亲委托机制。 JVM的 ClassLoader 采用的是树形结构,除了 根类加载器以外,每个ClassLoader 都会有 且仅有一个父类加载器,用户自定义的ClassLoader 默认的 父类加载器是系统类加载器,当然你可以自己指定需要用个ClassLoader 的实例,我们来看他 们的父子关系:
父类委托机制中,当一个java程序请求加载器 loader1 加载 Hello 类时, loader1 首先委托自己的父亲加载器加载 hello 类,若父亲加载器能加载,则由附加器完成加载人物,否则才由加载器 loader1 本身加载 Hello 类。下面我们来再次看一下 java 虚拟机自带的几个加载器:
除了java 虚拟机自带的加载器之外,我们用户自己也可以自定义自己的类加载器,根据自己的需要。。 Java 提供了抽象类 java.lang.ClassLoder ,所有用户自定义的类加载器都要继承这个 classloader 类。
注: 加载器之间的父子关系实际上指的是加载器对象之间的包装关系,而不是类之间的继承关系。一对父子加载器可能是同一个加载器类的两个实例,也可能不是。在子加载器对象中包装了一个父加载器对象. 当生成一个自定义的类加载器实例时,如果没有指定它的父加载器,那么系统类加载器就将成为该类加载器的父加载器。 如果在构造方法中指定父类加载器那么父类加载器就是指定的加载器。证明如下:
ClassLoader loader1 = new MyClassLoader(); //参数loader1将作为loader2的父加载器 ClassLoader loader2 = new MyClassLoader(loader1);
当Java 虚拟机要加载一个类时,到底该派哪个类加载器去加载呢 ?我们看下图:
Loader1和 loader2 是我们自己定义的两个类加载器, loader1 和 loader2 是父子关系。现在我们想让 loader2 这个类加载器加载我们自己写的一个 Sample 类: loader2.loadclass (“ sample ”),我们来分析一下看看到底应该用哪一个类加载器去加载。当这段代码被执行时, loader2 首先到自己的命名空间去查找 Sample 类是否已经被加载,如果被加载就直接返回这个类的 class 对象的引用。如果 Sample 类还没有被加载, loader2 首先请求 loader1 代为加载, loader1 再请求系统类加载器代为加载,系统类加载器再请求扩展类加载器,扩展类加载器再请求根类加载器,若根类加载器和扩展类加载器都不能加载,则系统类加载器尝试加载,若能加载,则将 Sample 类所对应的 Class 对象的引用返回给 loader1 , loader1 在将引用返回给 loader2 ,从而成功将 Sample 类加载到虚拟机。若系统类加载器不能加载 Sample 类,则 loader1 尝试加载 Sample 了哦,若 loader1 不能加载,则 loader2 尝试,若所有的类加载都不能加载,则抛出 ClassNotFoundException 异常。
定义类加载器 :如果某个类加载器能够加载一个类,那么该类加载器就称作:定义类加载器;定义类加载器及其所有子加载器都称作:初始类加载器
父委托机制的优点就是能够提高软件系统的安全性。因为在词机制下,用户自定义的类加载器不可能加载本应该由父加载器加载的可靠类,从而防止不可靠的恶意代码代替由父类加载器加载的可靠类,从而防止不可靠的甚至恶意的代码代替由父类加载器加载的可靠代码。如,java.lang.Object类总是由根类加载器加载的,其他任何用户自定义的类加载器都不可能加载含有恶意代码的java.lang.Object类。
命名空间,其实这里所说的命名空间就是我们 java中常用的package,每个类加载器都有自己的命名空间,命名空间由该加载器及所有父加载器所加载的类的组成。在同一个命名空间中,不会出现类的完整名字(包括类的包名)相同的两个雷;在不同的命名空间中,有可能会出现类的完整名字(包括类的包名)相同的两个类。
由同一类加载器加载的属于相同包的类组成了运行时包。决定两个类是不是属于同一个运行时包,不仅要看他们的包名称是否相同,还要看定义类加载器是否相同。只有属于同一运行时包的类之间才能相互访问可见(默认访问级别)的类和成员。假设用户自定义了一个类java.lang.TestCase并由用于自定义的类加载器加载,由于java.lang.TestCase和核心类库java.lang.*由不同的类加载器加载,他们属于不同的运行时包,所以java.lang.TestCase不能访问核心库java.lang包中的包可见成员。
参考资料:北京张龙老师免费培训视频《类加载器的父亲委托机制深度详解》
------------------------------------------------------------------------------------------------------------
《Java程序员由笨鸟到菜鸟》电子版书正式发布,欢迎大家下载
http://blog.csdn.net/csh624366188/article/details/7999247