zoukankan      html  css  js  c++  java
  • 深入理解java虚拟机(九)类加载器以及双亲委派模型

    虚拟机把类加载阶段中“通过一个类的全限定名来获取描述此类的二进制字节流”这个动作放到虚拟机外部去实现,以便让程序自己决定如何去获取所需要的类。实现这个动作的代码模块称为“类加载器”。

    类与类加载器

    任意一个类,都需要由加载它的类加载器和这个类本身共同确定其在Java 虚拟机中的唯一性,每一个类加载器,都拥有一个独立的类名称空间。这句话可以表达的更通俗一些:比较两个类是否相等,只有在这两个类是同一个类加载器加载的前提下才意义。

    这里的“相等”,包括代表类的 Class 对象的equals() 方法、isAssignableFrom() 方法、isInstance() 方法的返回结果,也包括 instanceof 关键字对对象所属关系判定等情况。下面代码演示了不同类加载器对 instanceof 关键字运算的结果的影响。

    [java] view plain copy
     
    1. public class ClassLoaderTest {  
    2.     public static void main(String[] args) throws Exception {  
    3.         ClassLoader myLoader = new ClassLoader() {  
    4.             @Override  
    5.             public Class<?> loadClass(String name)  
    6.                     throws ClassNotFoundException {  
    7.                 try {  
    8.                     String fileName = name.substring(name.lastIndexOf(".") + 1)  
    9.                             + ".class";  
    10.                     InputStream is = getClass().getResourceAsStream(fileName);  
    11.                     if (is == null) {  
    12.                         return super.loadClass(name);  
    13.                     }  
    14.                     byte[] b = new byte[is.available()];  
    15.                     is.read(b);  
    16.                     return defineClass(name, b, 0, b.length);  
    17.                 } catch (IOException e) {  
    18.                     throw new ClassNotFoundException(name);  
    19.                 }  
    20.             }  
    21.         };  
    22.   
    23.         Class c = myLoader.loadClass("org.bupt.xiaoye.blog.ClassLoaderTest");  
    24.         Object obj = c.newInstance();  
    25.         System.out.println(obj.getClass());  
    26.         System.out.println(ClassLoaderTest.class);  
    27.         System.out.println(obj instanceof ClassLoaderTest);  
    28.   
    29.     }  
    30. }  



    运行结果:

    [plain] view plain copy
     
    1. class org.bupt.xiaoye.blog.ClassLoaderTest  
    2. class org.bupt.xiaoye.blog.ClassLoaderTest  
    3. false  

    双亲委派模型

    从虚拟机的角度来讲,只存在两种不同的类加载器:

    一种是启动类加载器(Bootstrap ClassLoader),这个类加载器用 C++  语言实现, 是虚拟机自身的一部分:

    另一种就是所有其它的类加载器, 这些类加载器用Java 语言实现,独立于虚拟机外部,并且全都继承与抽象类 java.lang.ClassLoader。

    从 Java 开发人员的角度来看,类加载器还可以划分的更细致一些,绝大多数Java 程序都会用到以下3种系统提供的类加载器。

    • 启动类加载器(Bootstrap ClassLoader) : 这个类加载器负责将存放在 <JAVA_HOME>lib 目录中的,或者被 -Xbootclasspath 参数指定的路径中的,并且是虚拟机识别的(仅按照文件名识别,如rt.jar ,名字不符合类库不会加载) 类库加载到虚拟机内存中。启动类加载器无法被 java 程序直接引用,如需要,直接使用 null 代替即可。
    • 扩展类加载器(Extension ClassLoader):这个加载器由sun.misc.Launcher$ExtClassLoader 实现,它负责加载<JAVA_HOME>libext 目录中的,或者被 java.ext.dirs 系统变量所指定的路径中的所有类库,开发者可以直接使用扩展类加载器。
    • 应用程序类加载器(Application ClassLoader):这个类加载器由 sun.misc.Launcher$AppClassLoader 实现。这个这个类加载器是 ClassLoader 中的getSystemClassLoader() 方法的返回值,所以一般称它为系统类加载器。它负责加载用户路径(ClassPath)上所指定的类库,开发者可以使用这个类加载器,如果应用程序没有自定义过自己的类加载器,一般情况下这个就是程序中默认的类加载器。

    我们的应用程序都是由这3中类加载器互相配合进行加载的,如果有必要,还可以加入自己定义的类加载器。这些类加载器之间的关系一般如下图所示。“

    上图中的类加载器之间的这种层次关系,称为类加载器的双亲委派模型。双亲委派模型要求除了顶层的启动类加载器,其余的类加载器都应该有自己的父类加载器。这里类加载器之间的父子关系一般不会以继承关系来实现,而是使用组 载器的代码。

    双亲委托模型的工作过程是:如果一个类加载器收到了类加载器的请求,它首先不会自己尝试加载这个类,而是把这个请求委派给父类加载器去完成,没一个层次的类加载器都是如此,因此所有的加载请求最终都应该传送到顶层的启动类加载器中,只有当父类加载器反馈自己无法完成这个加载请求(它的搜说范围中没有找到所需的类时,子加载类才会尝试自己去加载)。

    使用双亲委派模型来组织类加载器之间的关系,有一个显而易见的好处是Java 类随着他的类加载器一起具备了一种优先级的层次关系。

    双亲委派模型对于保证Java 程序的稳定性很重要,但它的实现却非常简单,实现双亲委派的代码都集中在java.lang.ClassLoader 的 loadClass() 方法中:先检查类是否被家再过,若没有则调用父加载器的loadClass() 方法,若父加载器为空则默认使用启动类加载器作为父加载器。如果父加载器失败,抛出 ClassNotFoundException 异常后,再调用自己的 finClass() 方法进行加载。

    [java] view plain copy
     
    1. protected Class<?> loadClass(String name, boolean resolve)  
    2.         throws ClassNotFoundException {  
    3.     synchronized (getClassLoadingLock(name)) {  
    4.         // 首先检查类是否已经被家再过  
    5.         Class c = findLoadedClass(name);  
    6.         if (c == null) {  
    7.             long t0 = System.nanoTime();  
    8.             try {  
    9.                 if (parent != null) {  
    10.                     // 调用父加载器加载  
    11.                     c = parent.loadClass(name, false);  
    12.                 } else {  
    13.                     c = findBootstrapClassOrNull(name);  
    14.                 }  
    15.             } catch (ClassNotFoundException e) {  
    16.                 // ClassNotFoundException thrown if class not found  
    17.                 // from the non-null parent class loader  
    18.             }  
    19.   
    20.             if (c == null) {  
    21.                 // If still not found, then invoke findClass in order  
    22.                 // to find the class.  
    23.                 long t1 = System.nanoTime();  
    24.                 c = findClass(name);  
    25.   
    26.                 // this is the defining class loader; record the stats  
    27.                 sun.misc.PerfCounter.getParentDelegationTime().addTime(  
    28.                         t1 - t0);  
    29.                 sun.misc.PerfCounter.getFindClassTime().addElapsedTimeFrom(  
    30.                         t1);  
    31.                 sun.misc.PerfCounter.getFindClasses().increment();  
    32.             }  
    33.         }  
    34.         if (resolve) {  
    35.             resolveClass(c);  
    36.         }  
    37.         return c;  
    38.     }  
    39. }  



  • 相关阅读:
    对ManualResetEvent和AutoResetEvent的巩固练习
    经纬度点距离的那点儿事
    【读书笔记】C++Primer---第三章
    .NET应用程序调试—原理、工具、方法
    【读书笔记】C++Primer---第二章
    【读书笔记】C++Primer---第一章
    8 个最好的 jQuery 树形 Tree 插件
    C++中引用(&)的用法和应用实例
    自娱自乐之直接插入排序
    自娱自乐之堆排序
  • 原文地址:https://www.cnblogs.com/linghu-java/p/8554110.html
Copyright © 2011-2022 走看看