ClassLoader的双亲委派模型

类加载器用以实现类加载功能,并且确定被加载的类在Java虚拟机中的唯一性。

注意:由同一个类加载器加载,则这两个类相等,由不同的类加载器加载,则这两个类不相等。

我们知道类加载器主要分为:

  • 启动类加载器
  • 扩展类加载器
  • 应用程序类加载器

启动类加载器

负责加载以下类:

  • 存放在JAVA_HOME\lib目录中的类;
  • 被-Xbootclasspath参数所指定路径中、并且是被虚拟机识别的类库。

注意:启动类加载器不能直接被java程序直接饮用。

扩展类加载器

负责加载以下类:

  • JAVA_HOME\lib\ext目录中的类;
  • 被java.ext.dirs系统变量所指定的路径中的所有类。

注意:开发者可以直接使用扩展类加载器。

应用程序类加载器

负责加载用户类路径(classpath)上所指定的类库。

注意:该类加载器也被称为系统类加载器,开发者可以直接使用该类加载器。若开发者没有自定义类加载器,程序默认使用该类加载器。

各种类加载器并不是孤立的,而是互相配合使用。

在java虚拟机中,各种类加载器配合使用的模型就是双亲委派模型

扫描二维码关注公众号,回复: 5243762 查看本文章

双亲委派模型

双亲委派模型的工作流程全部在ClassLoader的loadClass()方法中执行:

@Override 
protected Class<?> loadClass(String name, boolean resolve) 
        throws ClassNotFoundException { 
    Class<?> c = findLoadedClass(name); 

  // 检查需要加载的类是否已经被加载过
    if (c == null) { 
        try { 
             // 若没有加载,则调用父加载器的loadClass()方法
            if (parent != null) { 
                c = parent.loadClass(name, false); 
            }else{ 
                // 若父类加载器为空,则默认使用启动类加载器作为父加载器
                c=findBootstrapClassOrNull(name); 
            } 
        } catch (ClassNotFoundException e) { 
            // 若父类加载器加载失败会抛出ClassNotFoundException, 
            //说明父类加载器无法完成加载请求 
        } 
        if(c==null){ 
            // 在父类加载器无法加载时 
            // 再调用本身的findClass方法进行类加载 
            c=findClass(name); 
        } 
    } 
    if(resolve){ 
        resolveClass(c); 
    } 
    return c; 
}
复制代码

若一个类加载器收到了类加载请求:

步骤:

  • 把该类加载请求委派给父加载器父完成,而不是自己去加载;(每层的类加载器都是如此,因此所有的类加载请求最终都会交由启动类加载器去加载)
  • 只用当父类加载器反馈自己无法完成该加载请求时,自加载器才会自己加载。

优点:

Java类随着它的类加载器一起具备了一种带优先级的层次关系。

猜你喜欢

转载自juejin.im/post/5c6d03a9f265da2d87636cb4
今日推荐