`

类加载器的双亲委派模型

    博客分类:
  • JVM
阅读更多

双亲委派模型
    从Java虚拟机的角度来讲,只存在两种不同的类加载器:一种是启动类加载器(Bootstrap ClassLoader),这个类加载器使用C++实现,时虚拟器自身的一部分;另一种就是所有其他的类加载器,这类加载器都由Java实现,独立于虚拟机外部,并且都继承自抽象类java.lang.ClassLoader。
    从开发人员的角度来看,类加载器还可以划分的更细致一些,绝大部分Java程序都会使用到一下3中系统提供的类加载器。
  • 启动类加载器(Bootstrap ClassLoader),这个类加载器前面已经介绍过,主要负责将存放在<JAVA_HOME>\lib目录中的,或者被-Xbootclasspath参数所指定的路径中的jar,并且是能够被虚拟机识别的(按照文件名识别,比如rt.jar,名字不符合虚拟机识别要求的即使放在该lib目录下也不会被加载)类库加载到虚拟机内存中。启动类加载器无法被Java程序直接引用,用户在编写自定义类加载器时,如果需要把加载请求委派给引导类加载器,那直接使用null代替即可。
  •    
  • 扩展类加载器(Extension ClassLoader):这个加载器由sun.misc.Launcher$ExtClassLoader实现,他负责加载<JAVA_HOME>/ext目录中的,或者被java.ext.dirs系统变量所指定的路径中的所有类库,开发者可以直接使用扩展类加载器。
  •    
  • 应用程序类加载器(Application ClassLoader):这个类加载器由sun.misc.Launcher$AppClassLoader实现。由于这个类加载器是ClassLoader中的getSystemClassLoader()方法的返回值,所以一般也称它为系统类加载器。他负责加载用户类路径(ClassPath)上所指定的类库,开发者可以直接使用这个类加载器器,如果应用程序中没有自定义过自己的类加载器,那么一般情况这个就是程序中默认的类加载器。
    我们的应用程序都是由这3中类加载器互相配合进行加载的,如果有必要,还可以加入我们自定义的类加载器,这些类加载器的关系如下图所示:

    上图中所展示的类加载器之间的层次关系,成为类加载器的双亲委派模型(Parents Delegation Model)。双亲委派模型要求除了顶层的启动类加载器外,其余的类加载器都应当有自己的父类加载器。但是这里类加载器之间的父子关系一般不会以继承的关系来实现,而是都是使用组合的方式来复用父类加载器的代码。
     类加载器的双亲委派模型在JDK1.2期间被引入并广泛应用于几乎所有的Java程序中,但它并不是一个强制性的约束模型,而是Java设计者推荐给开发者的一种类加载器实现方式。
     工作过程:如果一个类加载器收到了类加载的请求,他首先不会自己去尝试加载这个类,而是把这个请求委派给父类加载器去完成,每一个层次的类加载器如此,因此所有的加载请求最终都应该传送到顶层的启动类加载器中,只有当父加载器反馈自己无法完成这个加载请求(它的搜索范围中没有找到所需的类)时,子加载器才会尝试自己去加载。
      使用双亲委派模型的好处:Java类随着它的类加载器一起具备了一种带有优先级的层次关系。它存放在rt.jar中,无论哪一个类加载器都要加载这个类,最终都是委派给处于最顶端的启动类加载器进行加载,因此Object类在程序中的各种类加载环境中都是同一个类,相反,如果没有使用双亲委派模型,由各个类加载器自行去加载的话,(如果用户自己编写了一个不同的java.lang.1Object类,并放在classpath中,那么系统中将会出现多个不同的Object类,这样一来Java类型体系中最基础的行为也就无法保证,应用程序也将变得一片混乱),那么会使整个工程变得混乱。

      双亲委派模型对于保证Java程序的稳定运行很重要,但是它的实现却非常简单,实现双亲委派模型的代码都集中在java.lang.ClassLoader的loadClass()方法中。

     注意:即使是自定义了自己的类加载器,强行用defineClass()方法去加载一个以"java.lang"开头的类也不会成功。如果你尝试这么做,那么将会受到一个有虚拟机抛出的"java.lang.SecurityException:prohibited package name:java.lang"异常。
   
参考资料:
《深入理解Java虚拟机:JVM高级特性与最佳实践》 周志明著
http://www.360doc.com/content/14/0803/16/8072791_399149454.shtml
  • 大小: 49.5 KB
分享到:
评论

相关推荐

    类加载器和双亲委派模型加载类、类的加载优先级的详解.docx

    ### 类加载器与双亲委派模型详解 #### 类的生命周期与加载过程 类的生命周期主要包括七个阶段:加载、验证、准备、解析、初始化、使用和卸载。在这七个阶段中,验证、准备和解析统称为连接阶段。类的加载过程主要...

    第23讲请介绍类加载过程,什么是双亲委派模型1

    双亲委派模型是类加载机制的核心,它规定当类加载器尝试加载某个类型时,除非父加载器找不到该类型,否则任务会被委托给父加载器。这样可以避免类的重复加载,确保所有类都有统一的加载源,从而维护系统的一致性和...

    javakam#AndoJavaKotlin#类加载器及双亲委派1

    类加载器及双亲委派双亲委派模型的整个工作流程非常的简单,如下所示:如果一个类加载器收到了加载类的请求,它不会自己立即去加载类,它会先去请求父类加载器,每个层次的

    【JVM】类加载器与双亲委派模型

    类加载器按照特定的规则工作,其中最核心的就是双亲委派模型。 类加载器主要有三种类型: 1. 启动类加载器(Bootstrap ClassLoader):这是JVM内核的一部分,由C++实现,不继承自ClassLoader。它负责加载JVM的核心...

    细说Tomcat如何打破双亲委派(有源码和图)

    这是打破双亲委派模型的关键,因为WebAppClassLoader会优先尝试加载应用自己的类库,而不是依赖于全局的系统类加载器。 2. **CommonClassLoader**: 这是Tomcat的全局类加载器,负责加载服务器级别的类库,这些类库...

    Java类加载流程(双亲委派)流程图.zip

    双亲委派模型保证了Java核心库的稳定性,防止用户自定义类覆盖JDK内置类,同时使得系统能够共享公共类,避免类的重复加载。理解这一机制对于进行Java程序设计和优化至关重要,特别是在处理类的动态加载、插件系统和...

    java类加载器

    类加载器的设计遵循双亲委派模型,它分为三个主要部分:启动类加载器、扩展类加载器和应用类加载器。 #### 二、类加载过程 类加载过程主要包括三个步骤: 1. **加载**:通过类的全限定名找到该类的二进制字节流。...

    自定义类加载器实现自定义加载

    - Java中的类加载器采用双亲委派模型,即一个类首先由启动类加载器Bootstrap ClassLoader尝试加载,如果找不到则交给扩展类加载器Extension ClassLoader,再找不到则交由应用程序类加载器AppClassLoader,最后如果...

    JVM类加载器说明文档

    双亲委派模型的工作流程如下: 1. 当一个类加载器收到类加载请求时,它首先不会自己去尝试加载这个类,而是将这个请求委派给父类加载器去完成。 2. 如果父类加载器还不能加载,再由当前类加载器去尝试加载。 3. 如果...

    Java的类加载器

    加载器按照双亲委派模型工作,即从顶级的启动类加载器开始,逐级向下查找,直到找到目标类。 2. 类加载器类型: - 启动类加载器(Bootstrap ClassLoader):加载JDK核心类库,如rt.jar。 - 扩展类加载器...

    类加载器加载过程.rar

    Java类加载器遵循双亲委派模型,这意味着当一个类加载器收到加载类的请求时,它首先会把这个任务委托给它的父类加载器去完成。这个过程一直向上委托,直到Bootstrap ClassLoader(引导类加载器),如果Bootstrap ...

    Java类加载器原理

    类加载器遵循“双亲委派模型”(Delegation Model)。当一个类加载器接收到加载类的请求时,它首先会委托其父类加载器去尝试加载,如果父类加载器无法加载,再由当前加载器尝试。这个过程一直向上,直到Bootstrap类...

    Java类加载器机制与模型.pdf

    Java类加载器采用了**双亲委派模型(Parent Delegation Model)**,这意味着当一个类加载器收到加载类的请求时,它首先会委托其父类加载器去尝试加载,只有当父类加载器无法加载时,当前类加载器才会尝试自己加载。...

    类加载器(java)

    类加载器的工作机制遵循“双亲委派模型”:当一个类加载器接收到加载类的请求时,它首先会委托父加载器去尝试加载,只有在父加载器无法加载的情况下,子加载器才会尝试自己去加载。这种设计可以防止类的重复加载,并...

    tomcat类加载器

    类加载器遵循双亲委派模型,这意味着当一个类加载器尝试加载类时,它首先会将请求委托给其父类加载器,直到到达顶层的Bootstrap ClassLoader,如果父类加载器无法找到该类,子类加载器才会尝试自己加载。 在Tomcat...

    类加载器代码

    类加载器之间存在层次结构,当一个类加载器接收到加载类的请求时,它会先委托父类加载器去尝试加载,只有当父类加载器无法加载时,子类加载器才会尝试自己加载,这就是双亲委派模型。这种设计可以避免类的重复加载...

    java 详解类加载器的双亲委派及打破双亲委派

    在这个例子中,我们创建了一个名为`CustomClassLoader`的类加载器,它不再遵循双亲委派模型。当我们尝试加载类时,`loadClass`方法会直接调用`findClass`,而不是将任务委托给父类加载器。这样,我们可以在不依赖父...

Global site tag (gtag.js) - Google Analytics