`
m635674608
  • 浏览: 5028922 次
  • 性别: Icon_minigender_1
  • 来自: 南京
社区版块
存档分类
最新评论

jvm 类加载器 和 tomcat类加载器

    博客分类:
  • java
 
阅读更多



 类加载器基本概念

顾名思义,类加载器(class loader)用来加载 Java 类到 Java 虚拟机中。一般来说,Java 虚拟机使用 Java 类的方式如下:Java 源程序(.java 文件)在经过 Java 编译器编译之后就被转换成 Java 字节代码(.class 文件)。类加载器负责读取 Java 字节代码,并转换成 java.lang.Class类的一个实例。每个这样的实例用来表示一个 Java 类。通过此实例的 newInstance()方法就可以创建出该类的一个对象。实际的情况可能更加复杂,比如 Java 字节代码可能是通过工具动态生成的,也可能是通过网络下载的。

基本上所有的类加载器都是 java.lang.ClassLoader类的一个实例。下面详细介绍这个 Java 类。

java.lang.ClassLoader类介绍

java.lang.ClassLoader类的基本职责就是根据一个指定的类的名称,找到或者生成其对应的字节代码,然后从这些字节代码中定义出一个 Java 类,即 java.lang.Class类的一个实例。除此之外,ClassLoader还负责加载 Java 应用所需的资源,如图像文件和配置文件等。不过本文只讨论其加载类的功能。为了完成加载类的这个职责,ClassLoader提供了一系列的方法,比较重要的方法如 表 1所示。关于这些方法的细节会在下面进行介绍。


表 1. ClassLoader 中与加载类相关的方法

方法 说明
getParent() 返回该类加载器的父类加载器。
loadClass(String name) 加载名称为 name的类,返回的结果是 java.lang.Class类的实例。
findClass(String name) 查找名称为 name的类,返回的结果是 java.lang.Class类的实例。
findLoadedClass(String name) 查找名称为 name的已经被加载过的类,返回的结果是 java.lang.Class类的实例。
defineClass(String name, byte[] b, int off, int len) 把字节数组 b中的内容转换成 Java 类,返回的结果是 java.lang.Class类的实例。这个方法被声明为 final的。
resolveClass(Class<?> c) 链接指定的 Java 类。

对于 表 1中给出的方法,表示类名称的 name参数的值是类的二进制名称。需要注意的是内部类的表示,如 com.example.Sample$1com.example.Sample$Inner等表示方式。这些方法会在下面介绍类加载器的工作机制时,做进一步的说明。下面介绍类加载器的树状组织结构。

类加载器的树状组织结构

Java 中的类加载器大致可以分成两类,一类是系统提供的,另外一类则是由 Java 应用开发人员编写的。系统提供的类加载器主要有下面三个:

  • 引导类加载器(bootstrap class loader):它用来加载 Java 的核心库,是用原生代码来实现的,并不继承自 java.lang.ClassLoader
  • 扩展类加载器(extensions class loader):它用来加载 Java 的扩展库。Java 虚拟机的实现会提供一个扩展库目录。该类加载器在此目录里面查找并加载 Java 类。
  • 系统类加载器(system class loader):它根据 Java 应用的类路径(CLASSPATH)来加载 Java 类。一般来说,Java 应用的类都是由它来完成加载的。可以通过 ClassLoader.getSystemClassLoader()来获取它。

除了系统提供的类加载器以外,开发人员可以通过继承 java.lang.ClassLoader类的方式实现自己的类加载器,以满足一些特殊的需求。

除了引导类加载器之外,所有的类加载器都有一个父类加载器。通过 表 1中给出的 getParent()方法可以得到。对于系统提供的类加载器来说,系统类加载器的父类加载器是扩展类加载器,而扩展类加载器的父类加载器是引导类加载器;对于开发人员编写的类加载器来说,其父类加载器是加载此类加载器 Java 类的类加载器。因为类加载器 Java 类如同其它的 Java 类一样,也是要由类加载器来加载的。一般来说,开发人员编写的类加载器的父类加载器是系统类加载器。类加载器通过这种方式组织起来,形成树状结构。树的根节点就是引导类加载器。图 1中给出了一个典型的类加载器树状组织结构示意图,其中的箭头指向的是父类加载器。


图 1. 类加载器树状组织结构示意图
 

 

代码清单 1演示了类加载器的树状组织结构。


清单 1. 演示类加载器的树状组织结构

 public class ClassLoaderTree { 

    public static void main(String[] args) { 
        ClassLoader loader = ClassLoaderTree.class.getClassLoader(); 
        while (loader != null) { 
            System.out.println(loader.toString()); 
            loader = loader.getParent(); 
        } 
    } 
 }

 

每个 Java 类都维护着一个指向定义它的类加载器的引用,通过 getClassLoader()方法就可以获取到此引用。代码清单 1中通过递归调用 getParent()方法来输出全部的父类加载器。代码清单 1的运行结果如 代码清单 2所示。


清单 2. 演示类加载器的树状组织结构的运行结果

 sun.misc.Launcher$AppClassLoader@9304b1 
 sun.misc.Launcher$ExtClassLoader@190d11

 

代码清单 2所示,第一个输出的是 ClassLoaderTree类的类加载器,即系统类加载器。它是 sun.misc.Launcher$AppClassLoader类的实例;第二个输出的是扩展类加载器,是 sun.misc.Launcher$ExtClassLoader类的实例。需要注意的是这里并没有输出引导类加载器,这是由于有些 JDK 的实现对于父类加载器是引导类加载器的情况,getParent()方法返回 null

在了解了类加载器的树状组织结构之后,下面介绍类加载器的代理模式。

类加载器的代理模式

类加载器在尝试自己去查找某个类的字节代码并定义它时,会先代理给其父类加载器,由父类加载器先去尝试加载这个类,依次类推。在介绍代理模式背后的动机之前,首先需要说明一下 Java 虚拟机是如何判定两个 Java 类是相同的。Java 虚拟机不仅要看类的全名是否相同,还要看加载此类的类加载器是否一样。只有两者都相同的情况,才认为两个类是相同的。即便是同样的字节代码,被不同的类加载器加载之后所得到的类,也是不同的。比如一个 Java 类 com.example.Sample,编译之后生成了字节代码文件 Sample.class。两个不同的类加载器 ClassLoaderAClassLoaderB分别读取了这个 Sample.class文件,并定义出两个 java.lang.Class类的实例来表示这个类。这两个实例是不相同的。对于 Java 虚拟机来说,它们是不同的类。试图对这两个类的对象进行相互赋值,会抛出运行时异常 ClassCastException。下面通过示例来具体说明。代码清单 3中给出了 Java 类 com.example.Sample


清单 3. com.example.Sample 类

 package com.example; 

 public class Sample { 
    private Sample instance; 

    public void setSample(Object instance) { 
        this.instance = (Sample) instance; 
    } 
 }

 

代码清单 3所示,com.example.Sample类的方法 setSample接受一个 java.lang.Object类型的参数,并且会把该参数强制转换成 com.example.Sample类型。测试 Java 类是否相同的代码如 代码清单 4所示。


清单 4. 测试 Java 类是否相同

 public void testClassIdentity() { 
    String classDataRootPath = "C:\\workspace\\Classloader\\classData"; 
    FileSystemClassLoader fscl1 = new FileSystemClassLoader(classDataRootPath); 
    FileSystemClassLoader fscl2 = new FileSystemClassLoader(classDataRootPath); 
    String className = "com.example.Sample"; 	
    try { 
        Class<?> class1 = fscl1.loadClass(className); 
        Object obj1 = class1.newInstance(); 
        Class<?> class2 = fscl2.loadClass(className); 
        Object obj2 = class2.newInstance(); 
        Method setSampleMethod = class1.getMethod("setSample", java.lang.Object.class); 
        setSampleMethod.invoke(obj1, obj2); 
    } catch (Exception e) { 
        e.printStackTrace(); 
    } 
 }

 

代码清单 4中使用了类 FileSystemClassLoader的两个不同实例来分别加载类 com.example.Sample,得到了两个不同的 java.lang.Class的实例,接着通过 newInstance()方法分别生成了两个类的对象 obj1obj2,最后通过 Java 的反射 API 在对象 obj1上调用方法 setSample,试图把对象 obj2赋值给 obj1内部的 instance对象。代码清单 4的运行结果如 代码清单 5所示。


清单 5. 测试 Java 类是否相同的运行结果

java.lang.reflect.InvocationTargetException 
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method) 
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:39) 
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:25)
at java.lang.reflect.Method.invoke(Method.java:597) 
at classloader.ClassIdentity.testClassIdentity(ClassIdentity.java:26) 
at classloader.ClassIdentity.main(ClassIdentity.java:9) 
Caused by: java.lang.ClassCastException: com.example.Sample 
cannot be cast to com.example.Sample 
at com.example.Sample.setSample(Sample.java:7) 
... 6 more 

 

代码清单 5给出的运行结果可以看到,运行时抛出了 java.lang.ClassCastException异常。虽然两个对象 obj1obj2的类的名字相同,但是这两个类是由不同的类加载器实例来加载的,因此不被 Java 虚拟机认为是相同的。

了解了这一点之后,就可以理解代理模式的设计动机了。代理模式是为了保证 Java 核心库的类型安全。所有 Java 应用都至少需要引用 java.lang.Object类,也就是说在运行的时候,java.lang.Object这个类需要被加载到 Java 虚拟机中。如果这个加载过程由 Java 应用自己的类加载器来完成的话,很可能就存在多个版本的 java.lang.Object类,而且这些类之间是不兼容的。通过代理模式,对于 Java 核心库的类的加载工作由引导类加载器来统一完成,保证了 Java 应用所使用的都是同一个版本的 Java 核心库的类,是互相兼容的。

不同的类加载器为相同名称的类创建了额外的名称空间。相同名称的类可以并存在 Java 虚拟机中,只需要用不同的类加载器来加载它们即可。不同类加载器加载的类之间是不兼容的,这就相当于在 Java 虚拟机内部创建了一个个相互隔离的 Java 类空间。这种技术在许多框架中都被用到,后面会详细介绍。

下面具体介绍类加载器加载类的详细过程。

加载类的过程

在前面介绍类加载器的代理模式的时候,提到过类加载器会首先代理给其它类加载器来尝试加载某个类。这就意味着真正完成类的加载工作的类加载器和启动这个加载过程的类加载器,有可能不是同一个。真正完成类的加载工作是通过调用 defineClass来实现的;而启动类的加载过程是通过调用 loadClass来实现的。前者称为一个类的定义加载器(defining loader),后者称为初始加载器(initiating loader)。在 Java 虚拟机判断两个类是否相同的时候,使用的是类的定义加载器。也就是说,哪个类加载器启动类的加载过程并不重要,重要的是最终定义这个类的加载器。两种类加载器的关联之处在于:一个类的定义加载器是它引用的其它类的初始加载器。如类 com.example.Outer引用了类 com.example.Inner,则由类 com.example.Outer的定义加载器负责启动类 com.example.Inner的加载过程。

方法 loadClass()抛出的是 java.lang.ClassNotFoundException异常;方法 defineClass()抛出的是 java.lang.NoClassDefFoundError异常。

类加载器在成功加载某个类之后,会把得到的 java.lang.Class类的实例缓存起来。下次再请求加载该类的时候,类加载器会直接使用缓存的类的实例,而不会尝试再次加载。也就是说,对于一个类加载器实例来说,相同全名的类只加载一次,即 loadClass方法不会被重复调用。

下面讨论另外一种类加载器:线程上下文类加载器。

线程上下文类加载器

线程上下文类加载器(context class loader)是从 JDK 1.2 开始引入的。类 java.lang.Thread中的方法 getContextClassLoader()setContextClassLoader(ClassLoader cl)用来获取和设置线程的上下文类加载器。如果没有通过 setContextClassLoader(ClassLoader cl)方法进行设置的话,线程将继承其父线程的上下文类加载器。Java 应用运行的初始线程的上下文类加载器是系统类加载器。在线程中运行的代码可以通过此类加载器来加载类和资源。

前面提到的类加载器的代理模式并不能解决 Java 应用开发中会遇到的类加载器的全部问题。Java 提供了很多服务提供者接口(Service Provider Interface,SPI),允许第三方为这些接口提供实现。常见的 SPI 有 JDBC、JCE、JNDI、JAXP 和 JBI 等。这些 SPI 的接口由 Java 核心库来提供,如 JAXP 的 SPI 接口定义包含在 javax.xml.parsers包中。这些 SPI 的实现代码很可能是作为 Java 应用所依赖的 jar 包被包含进来,可以通过类路径(CLASSPATH)来找到,如实现了 JAXP SPI 的 Apache Xerces所包含的 jar 包。SPI 接口中的代码经常需要加载具体的实现类。如 JAXP 中的 javax.xml.parsers.DocumentBuilderFactory类中的 newInstance()方法用来生成一个新的 DocumentBuilderFactory的实例。这里的实例的真正的类是继承自 javax.xml.parsers.DocumentBuilderFactory,由 SPI 的实现所提供的。如在 Apache Xerces 中,实现的类是 org.apache.xerces.jaxp.DocumentBuilderFactoryImpl。而问题在于,SPI 的接口是 Java 核心库的一部分,是由引导类加载器来加载的;SPI 实现的 Java 类一般是由系统类加载器来加载的。引导类加载器是无法找到 SPI 的实现类的,因为它只加载 Java 的核心库。它也不能代理给系统类加载器,因为它是系统类加载器的祖先类加载器。也就是说,类加载器的代理模式无法解决这个问题。

线程上下文类加载器正好解决了这个问题。如果不做任何的设置,Java 应用的线程的上下文类加载器默认就是系统上下文类加载器。在 SPI 接口的代码中使用线程上下文类加载器,就可以成功的加载到 SPI 实现的类。线程上下文类加载器在很多 SPI 的实现中都会用到。

下面介绍另外一种加载类的方法:Class.forName

Class.forName

Class.forName是一个静态方法,同样可以用来加载类。该方法有两种形式:Class.forName(String name, boolean initialize, ClassLoader loader)Class.forName(String className)。第一种形式的参数 name表示的是类的全名;initialize表示是否初始化类;loader表示加载时使用的类加载器。第二种形式则相当于设置了参数 initialize的值为 trueloader的值为当前类的类加载器。Class.forName的一个很常见的用法是在加载数据库驱动的时候。如 Class.forName("org.apache.derby.jdbc.EmbeddedDriver").newInstance()用来加载 Apache Derby 数据库的驱动。

在介绍完类加载器相关的基本概念之后,下面介绍如何开发自己的类加载器。


开发自己的类加载器

虽然在绝大多数情况下,系统默认提供的类加载器实现已经可以满足需求。但是在某些情况下,您还是需要为应用开发出自己的类加载器。比如您的应用通过网络来传输 Java 类的字节代码,为了保证安全性,这些字节代码经过了加密处理。这个时候您就需要自己的类加载器来从某个网络地址上读取加密后的字节代码,接着进行解密和验证,最后定义出要在 Java 虚拟机中运行的类来。下面将通过两个具体的实例来说明类加载器的开发。

文件系统类加载器

第一个类加载器用来加载存储在文件系统上的 Java 字节代码。完整的实现如 代码清单 6所示。


清单 6. 文件系统类加载器

 public class FileSystemClassLoader extends ClassLoader { 

    private String rootDir; 

    public FileSystemClassLoader(String rootDir) { 
        this.rootDir = rootDir; 
    } 

    protected Class<?> findClass(String name) throws ClassNotFoundException { 
        byte[] classData = getClassData(name); 
        if (classData == null) { 
            throw new ClassNotFoundException(); 
        } 
        else { 
            return defineClass(name, classData, 0, classData.length); 
        } 
    } 

    private byte[] getClassData(String className) { 
        String path = classNameToPath(className); 
        try { 
            InputStream ins = new FileInputStream(path); 
            ByteArrayOutputStream baos = new ByteArrayOutputStream(); 
            int bufferSize = 4096; 
            byte[] buffer = new byte[bufferSize]; 
            int bytesNumRead = 0; 
            while ((bytesNumRead = ins.read(buffer)) != -1) { 
                baos.write(buffer, 0, bytesNumRead); 
            } 
            return baos.toByteArray(); 
        } catch (IOException e) { 
            e.printStackTrace(); 
        } 
        return null; 
    } 

    private String classNameToPath(String className) { 
        return rootDir + File.separatorChar 
                + className.replace('.', File.separatorChar) + ".class"; 
    } 
 }

 

代码清单 6所示,类 FileSystemClassLoader继承自类 java.lang.ClassLoader。在 表 1中列出的 java.lang.ClassLoader类的常用方法中,一般来说,自己开发的类加载器只需要覆写 findClass(String name)方法即可。java.lang.ClassLoader类的方法 loadClass()封装了前面提到的代理模式的实现。该方法会首先调用 findLoadedClass()方法来检查该类是否已经被加载过;如果没有加载过的话,会调用父类加载器的 loadClass()方法来尝试加载该类;如果父类加载器无法加载该类的话,就调用 findClass()方法来查找该类。因此,为了保证类加载器都正确实现代理模式,在开发自己的类加载器时,最好不要覆写 loadClass()方法,而是覆写 findClass()方法。

FileSystemClassLoaderfindClass()方法首先根据类的全名在硬盘上查找类的字节代码文件(.class 文件),然后读取该文件内容,最后通过 defineClass()方法来把这些字节代码转换成 java.lang.Class类的实例。

网络类加载器

下面将通过一个网络类加载器来说明如何通过类加载器来实现组件的动态更新。即基本的场景是:Java 字节代码(.class)文件存放在服务器上,客户端通过网络的方式获取字节代码并执行。当有版本更新的时候,只需要替换掉服务器上保存的文件即可。通过类加载器可以比较简单的实现这种需求。

NetworkClassLoader负责通过网络下载 Java 类字节代码并定义出 Java 类。它的实现与 FileSystemClassLoader类似。在通过 NetworkClassLoader加载了某个版本的类之后,一般有两种做法来使用它。第一种做法是使用 Java 反射 API。另外一种做法是使用接口。需要注意的是,并不能直接在客户端代码中引用从服务器上下载的类,因为客户端代码的类加载器找不到这些类。使用 Java 反射 API 可以直接调用 Java 类的方法。而使用接口的做法则是把接口的类放在客户端中,从服务器上加载实现此接口的不同版本的类。在客户端通过相同的接口来使用这些实现类。网络类加载器的具体代码见 下载

在介绍完如何开发自己的类加载器之后,下面说明类加载器和 Web 容器的关系。


类加载器与 Web 容器

对于运行在 Java EE™容器中的 Web 应用来说,类加载器的实现方式与一般的 Java 应用有所不同。不同的 Web 容器的实现方式也会有所不同。以 Apache Tomcat 来说,每个 Web 应用都有一个对应的类加载器实例。该类加载器也使用代理模式,所不同的是它是首先尝试去加载某个类,如果找不到再代理给父类加载器。这与一般类加载器的顺序是相反的。这是 Java Servlet 规范中的推荐做法,其目的是使得 Web 应用自己的类的优先级高于 Web 容器提供的类。这种代理模式的一个例外是:Java 核心库的类是不在查找范围之内的。这也是为了保证 Java 核心库的类型安全。

绝大多数情况下,Web 应用的开发人员不需要考虑与类加载器相关的细节。下面给出几条简单的原则:

  • 每个 Web 应用自己的 Java 类文件和使用的库的 jar 包,分别放在 WEB-INF/classesWEB-INF/lib目录下面。
  • 多个应用共享的 Java 类文件和 jar 包,分别放在 Web 容器指定的由所有 Web 应用共享的目录下面。
  • 当出现找不到类的错误时,检查当前类的类加载器和当前线程的上下文类加载器是否正确。

在介绍完类加载器与 Web 容器的关系之后,下面介绍它与 OSGi 的关系。


类加载器与 OSGi

OSGi™是 Java 上的动态模块系统。它为开发人员提供了面向服务和基于组件的运行环境,并提供标准的方式用来管理软件的生命周期。OSGi 已经被实现和部署在很多产品上,在开源社区也得到了广泛的支持。Eclipse 就是基于 OSGi 技术来构建的。

OSGi 中的每个模块(bundle)都包含 Java 包和类。模块可以声明它所依赖的需要导入(import)的其它模块的 Java 包和类(通过 Import-Package),也可以声明导出(export)自己的包和类,供其它模块使用(通过 Export-Package)。也就是说需要能够隐藏和共享一个模块中的某些 Java 包和类。这是通过 OSGi 特有的类加载器机制来实现的。OSGi 中的每个模块都有对应的一个类加载器。它负责加载模块自己包含的 Java 包和类。当它需要加载 Java 核心库的类时(以 java开头的包和类),它会代理给父类加载器(通常是启动类加载器)来完成。当它需要加载所导入的 Java 类时,它会代理给导出此 Java 类的模块来完成加载。模块也可以显式的声明某些 Java 包和类,必须由父类加载器来加载。只需要设置系统属性 org.osgi.framework.bootdelegation的值即可。

假设有两个模块 bundleA 和 bundleB,它们都有自己对应的类加载器 classLoaderA 和 classLoaderB。在 bundleA 中包含类 com.bundleA.Sample,并且该类被声明为导出的,也就是说可以被其它模块所使用的。bundleB 声明了导入 bundleA 提供的类 com.bundleA.Sample,并包含一个类 com.bundleB.NewSample继承自 com.bundleA.Sample。在 bundleB 启动的时候,其类加载器 classLoaderB 需要加载类 com.bundleB.NewSample,进而需要加载类 com.bundleA.Sample。由于 bundleB 声明了类 com.bundleA.Sample是导入的,classLoaderB 把加载类 com.bundleA.Sample的工作代理给导出该类的 bundleA 的类加载器 classLoaderA。classLoaderA 在其模块内部查找类 com.bundleA.Sample并定义它,所得到的类 com.bundleA.Sample实例就可以被所有声明导入了此类的模块使用。对于以 java开头的类,都是由父类加载器来加载的。如果声明了系统属性 org.osgi.framework.bootdelegation=com.example.core.*,那么对于包 com.example.core中的类,都是由父类加载器来完成的。

OSGi 模块的这种类加载器结构,使得一个类的不同版本可以共存在 Java 虚拟机中,带来了很大的灵活性。不过它的这种不同,也会给开发人员带来一些麻烦,尤其当模块需要使用第三方提供的库的时候。下面提供几条比较好的建议:

  • 如果一个类库只有一个模块使用,把该类库的 jar 包放在模块中,在 Bundle-ClassPath中指明即可。
  • 如果一个类库被多个模块共用,可以为这个类库单独的创建一个模块,把其它模块需要用到的 Java 包声明为导出的。其它模块声明导入这些类。
  • 如果类库提供了 SPI 接口,并且利用线程上下文类加载器来加载 SPI 实现的 Java 类,有可能会找不到 Java 类。如果出现了 NoClassDefFoundError异常,首先检查当前线程的上下文类加载器是否正确。通过 Thread.currentThread().getContextClassLoader()就可以得到该类加载器。该类加载器应该是该模块对应的类加载器。如果不是的话,可以首先通过 class.getClassLoader()来得到模块对应的类加载器,再通过 Thread.currentThread().setContextClassLoader()来设置当前线程的上下文类加载器。

总结

类加载器是 Java 语言的一个创新。它使得动态安装和更新软件组件成为可能。本文详细介绍了类加载器的相关话题,包括基本概念、代理模式、线程上下文类加载器、与 Web 容器和 OSGi 的关系等。开发人员在遇到 ClassNotFoundExceptionNoClassDefFoundError等异常的时候,应该检查抛出异常的类的类加载器和当前线程的上下文类加载器,从中可以发现问题的所在。在开发自己的类加载器的时候,需要注意与已有的类加载器组织结构的协调。

<!-- CMA ID: 470295 --><!-- Site ID: 10 --><!-- XSLT stylesheet used to transform this file: dw-document-html-6.0.xsl -->

 

tomcat  WebappClassLoader加载器

    //要先加载相关实体资源(.jar) 再加载查找的资源本身
    protected ResourceEntry findResourceInternal(String name, String path) {
        //先根据类名从缓存中查找对应资源 ,有则直接返回 
        ResourceEntry entry = (ResourceEntry) resourceEntries.get(name);
        if (entry != null)
            return entry;
        int contentLength = -1;//资源二进制数据长度
        InputStream binaryStream = null;//资源二进制输入流

        int jarFilesLength = jarFiles.length;//classpath中的jar包个数
        int repositoriesLength = repositories.length;//仓库数(classpath每一段称为repository仓库)
        int i;
        Resource resource = null;//加载的资源实体
        boolean fileNeedConvert = false;
        //对每个仓库迭代,直接找到相应的entry,如果查找的资源是一个独立的文件,在这个代码块可以查找到相应资源,
        //如果是包含在jar包中的类,这段代码并不能找出其对应的资源
        for (i = 0; (entry == null) && (i < repositoriesLength); i++) {
            try {
                String fullPath = repositories[i] + path;//仓库路径 加资源路径得到全路径
                Object lookupResult = resources.lookup(fullPath);//从资源库中查找资源

                if (lookupResult instanceof Resource) {
                    resource = (Resource) lookupResult;
                }
                //到这里没有抛出异常,说明资源已经找到,现在构造entry对象
                 if (securityManager != null) {
                    PrivilegedAction dp =
                        new PrivilegedFindResource(files[i], path);
                    entry = (ResourceEntry)AccessController.doPrivileged(dp);
                 } else {
                    entry = findResourceInternal(files[i], path);//这个方式只是构造entry并给其codebase和source赋值
                 }
                 //获取资源长度和最后修改时间
                ResourceAttributes attributes =
                    (ResourceAttributes) resources.getAttributes(fullPath);
                contentLength = (int) attributes.getContentLength();
                entry.lastModified = attributes.getLastModified();
                //资源找到,将二进制输入流赋给binaryStream
                if (resource != null) {
                    try {
                        binaryStream = resource.streamContent();
                    } catch (IOException e) {
                        return null;
                    }
                    //将资源的最后修改时间加到列表中去,代码略去,参加源码 

                }

            } catch (NamingException e) {
            }
        }
        if ((entry == null) && (notFoundResources.containsKey(name)))
            return null;
        //开始从jar包中查找
        JarEntry jarEntry = null;
        synchronized (jarFiles) {
            if (!openJARs()) {
                return null;
            }
            for (i = 0; (entry == null) && (i < jarFilesLength); i++) {
                jarEntry = jarFiles[i].getJarEntry(path);//根据路径从jar包中查找资源
                //如果jar包中找到资源,则定义entry并将二进制流等数据赋给entry,同时将jar包解压到workdir.
                if (jarEntry != null) {
                    entry = new ResourceEntry();
                    try {
                        entry.codeBase = getURL(jarRealFiles[i], false);
                        String jarFakeUrl = getURI(jarRealFiles[i]).toString();
                        jarFakeUrl = "jar:" + jarFakeUrl + "!/" + path;
                        entry.source = new URL(jarFakeUrl);
                        entry.lastModified = jarRealFiles[i].lastModified();
                    } catch (MalformedURLException e) {
                        return null;
                    }
                    contentLength = (int) jarEntry.getSize();
                    try {
                        entry.manifest = jarFiles[i].getManifest();
                        binaryStream = jarFiles[i].getInputStream(jarEntry);
                    } catch (IOException e) {
                        return null;
                    }
                    if (antiJARLocking && !(path.endsWith(".class"))) {
                        //解压jar包代码,参见源码
                    }
                }
            }
            if (entry == null) {
                synchronized (notFoundResources) {
                    notFoundResources.put(name, name);
                }
                return null;
            }
            //从二进制流将资源内容读出
            if (binaryStream != null) {
                byte[] binaryContent = new byte[contentLength];
                //读二进制流的代码,这里省去,参见源码
                entry.binaryContent = binaryContent;
            }
        }
        // 将资源加到缓存中
        synchronized (resourceEntries) {
            ResourceEntry entry2 = (ResourceEntry) resourceEntries.get(name);
            if (entry2 == null) {
                resourceEntries.put(name, entry);
            } else {
                entry = entry2;
            }
        }
        return entry;
    }

 

Tomcat负责Web应用的类加载的是org.apache.catalina.loader.WebappClassLoader,它有几个比较重要的方法:findClass(),loadClass(),findClassInternal(),findResourceInternal().
 
   WebappClassLoader类加载器被用来加载一个类的时候,loadClass()会被调用,loadClass()则调用findClass()。后两个方法是WebappClassLoader的私有方法,findClass()调用 findClassInternal()来创建class对象,而findClassInternal()则需要 findResourceInternal()来查找.class文件。
 
通常自己实现类记载器只要实现findclass即可,这里为了实现特殊目的而override了loadClass().
 
下面是精简过的代码(去除了几乎全部关于log、异常和安全控制的代码):
Java代码 复制代码 收藏代码
  1. public Class loadClass(String name, boolean resolve)   
  2.         throws ClassNotFoundException {   
  3.     Class clazz = null;   
  4.     // (0) 先从自己的缓存中查找,有则返回,无则继续   
  5.     clazz = findLoadedClass0(name);   
  6.     if (clazz != null) {   
  7.         if (resolve)   
  8.             resolveClass(clazz);   
  9.         return (clazz);   
  10.     }   
  11.     // (0.1) 再从parent的缓存中查找   
  12.     clazz = findLoadedClass(name);   
  13.     if (clazz != null) {   
  14.         if (resolve)   
  15.             resolveClass(clazz);   
  16.         return (clazz);   
  17.     }   
  18.     // (0.2) 缓存中没有,则首先使用system类加载器来加载   
  19.     clazz = system.loadClass(name);   
  20.     if (clazz != null) {   
  21.         if (resolve)   
  22.             resolveClass(clazz);   
  23.         return (clazz);   
  24.     }   
  25.     // 判断是否需要先让parent代理   
  26.     boolean delegateLoad = delegate || filter(name);   
  27.     // (1) 先让parent加载,通常delegateLoad == false,即这一步不会执行   
  28.     if (delegateLoad) {   
  29.         ClassLoader loader = parent;   
  30.         if (loader == null)   
  31.             loader = system;   
  32.         clazz = loader.loadClass(name);   
  33.         if (clazz == null) {   
  34.             if (resolve)   
  35.                 resolveClass(clazz);   
  36.             return (clazz);   
  37.         }   
  38.     }   
  39.     // (2) delegateLoad == false 或者 parent加载失败,调用自身的加载机制   
  40.     clazz = findClass(name);   
  41.     if (clazz == null) {   
  42.         if (resolve)   
  43.             resolveClass(clazz);   
  44.         return (clazz);   
  45.     }   
  46.     // (3) 自己加载失败,则请求parent代理加载   
  47.     if (!delegateLoad) {   
  48.         ClassLoader loader = parent;   
  49.         if (loader == null)   
  50.             loader = system;   
  51.         clazz = loader.loadClass(name);   
  52.         if (clazz == null) {   
  53.             return (clazz);   
  54.         }   
  55.     }   
  56.     throw new ClassNotFoundException(name);   
  57. }  
	public Class loadClass(String name, boolean resolve)
			throws ClassNotFoundException {
		Class clazz = null;
		// (0) 先从自己的缓存中查找,有则返回,无则继续
		clazz = findLoadedClass0(name);
		if (clazz != null) {
			if (resolve)
				resolveClass(clazz);
			return (clazz);
		}
		// (0.1) 再从parent的缓存中查找
		clazz = findLoadedClass(name);
		if (clazz != null) {
			if (resolve)
				resolveClass(clazz);
			return (clazz);
		}
		// (0.2) 缓存中没有,则首先使用system类加载器来加载
		clazz = system.loadClass(name);
		if (clazz != null) {
			if (resolve)
				resolveClass(clazz);
			return (clazz);
		}
		// 判断是否需要先让parent代理
		boolean delegateLoad = delegate || filter(name);
		// (1) 先让parent加载,通常delegateLoad == false,即这一步不会执行
		if (delegateLoad) {
			ClassLoader loader = parent;
			if (loader == null)
				loader = system;
			clazz = loader.loadClass(name);
			if (clazz == null) {
				if (resolve)
					resolveClass(clazz);
				return (clazz);
			}
		}
		// (2) delegateLoad == false 或者 parent加载失败,调用自身的加载机制
		clazz = findClass(name);
		if (clazz == null) {
			if (resolve)
				resolveClass(clazz);
			return (clazz);
		}
		// (3) 自己加载失败,则请求parent代理加载
		if (!delegateLoad) {
			ClassLoader loader = parent;
			if (loader == null)
				loader = system;
			clazz = loader.loadClass(name);
			if (clazz == null) {
				return (clazz);
			}
		}
		throw new ClassNotFoundException(name);
	}

 

Java代码 复制代码 收藏代码
  1. public Class findClass(String name) throws ClassNotFoundException {   
  2.     // 先试图自己加载类,找不到则请求parent来加载   
  3.     // 注意这点和java默认的双亲委托模式不同   
  4.     Class clazz = null;   
  5.     clazz = findClassInternal(name);   
  6.     if ((clazz == null) && hasExternalRepositories) {   
  7.         synchronized (this) {   
  8.             clazz = super.findClass(name);   
  9.         }   
  10.     }   
  11.     if (clazz == null) {   
  12.         throw new ClassNotFoundException(name);   
  13.     }   
  14.     return (clazz);   
  15. }  
	public Class findClass(String name) throws ClassNotFoundException {
		// 先试图自己加载类,找不到则请求parent来加载
		// 注意这点和java默认的双亲委托模式不同
		Class clazz = null;
		clazz = findClassInternal(name);
		if ((clazz == null) && hasExternalRepositories) {
			synchronized (this) {
				clazz = super.findClass(name);
			}
		}
		if (clazz == null) {
			throw new ClassNotFoundException(name);
		}
		return (clazz);
	}

 

Java代码 复制代码 收藏代码
  1. protected Class findClassInternal(String name) throws ClassNotFoundException {   
  2.     if (!validate(name))   
  3.         throw new ClassNotFoundException(name);   
  4.     // 根据类名查找资源   
  5.     String tempPath = name.replace('.''/');   
  6.     String classPath = tempPath + ".class";   
  7.     ResourceEntry entry = null;   
  8.     entry = findResourceInternal(name, classPath);   
  9.   
  10.     if (entry == null)   
  11.         throw new ClassNotFoundException(name);   
  12.     // 如果以前已经加载成功过这个类,直接返回   
  13.   
  14.     Class clazz = entry.loadedClass;   
  15.     if (clazz != null)   
  16.         return clazz;   
  17.     // 以下根据找到的资源(.class文件)进行:1、定义package;2、对package安全检查;3、定义class,即创建class对象   
  18.     synchronized (this) {   
  19.         if (entry.binaryContent == null && entry.loadedClass == null)   
  20.             throw new ClassNotFoundException(name);   
  21.         // Looking up the package   
  22.         String packageName = null;   
  23.         int pos = name.lastIndexOf('.');   
  24.         if (pos != -1)   
  25.             packageName = name.substring(0, pos);   
  26.         Package pkg = null;   
  27.         if (packageName != null) {   
  28.             pkg = getPackage(packageName);   
  29.             // Define the package (if null)   
  30.             if (pkg == null) {   
  31.                 // 定义package的操作,此处省略,具体参看源码   
  32.                 pkg = getPackage(packageName);   
  33.             }   
  34.         }   
  35.         if (securityManager != null) {   
  36.             // 安全检查操作,此处省略,具体参看源码   
  37.         }   
  38.         // 创建class对象并返回   
  39.         if (entry.loadedClass == null) {   
  40.             try {   
  41.                 clazz = defineClass(name, entry.binaryContent, 0, entry.binaryContent.length, new CodeSource(entry.codeBase, entry.certificates));   
  42.             } catch (UnsupportedClassVersionError ucve) {   
  43.                 throw new UnsupportedClassVersionError(ucve.getLocalizedMessage() + " " + sm.getString("webappClassLoader.wrongVersion", name));   
  44.             }   
  45.             entry.loadedClass = clazz;   
  46.             entry.binaryContent = null;   
  47.             entry.source = null;   
  48.             entry.codeBase = null;   
  49.             entry.manifest = null;   
  50.             entry.certificates = null;   
  51.         } else {   
  52.             clazz = entry.loadedClass;   
  53.         }   
  54.     }   
  55.     return clazz;   
  56. }  
	protected Class findClassInternal(String name) throws ClassNotFoundException {
		if (!validate(name))
			throw new ClassNotFoundException(name);
		// 根据类名查找资源
		String tempPath = name.replace('.', '/');
		String classPath = tempPath + ".class";
		ResourceEntry entry = null;
		entry = findResourceInternal(name, classPath);

		if (entry == null)
			throw new ClassNotFoundException(name);
		// 如果以前已经加载成功过这个类,直接返回

		Class clazz = entry.loadedClass;
		if (clazz != null)
			return clazz;
		// 以下根据找到的资源(.class文件)进行:1、定义package;2、对package安全检查;3、定义class,即创建class对象
		synchronized (this) {
			if (entry.binaryContent == null && entry.loadedClass == null)
				throw new ClassNotFoundException(name);
			// Looking up the package
			String packageName = null;
			int pos = name.lastIndexOf('.');
			if (pos != -1)
				packageName = name.substring(0, pos);
			Package pkg = null;
			if (packageName != null) {
				pkg = getPackage(packageName);
				// Define the package (if null)
				if (pkg == null) {
					// 定义package的操作,此处省略,具体参看源码
					pkg = getPackage(packageName);
				}
			}
			if (securityManager != null) {
				// 安全检查操作,此处省略,具体参看源码
			}
			// 创建class对象并返回
			if (entry.loadedClass == null) {
				try {
					clazz = defineClass(name, entry.binaryContent, 0, entry.binaryContent.length, new CodeSource(entry.codeBase, entry.certificates));
				} catch (UnsupportedClassVersionError ucve) {
					throw new UnsupportedClassVersionError(ucve.getLocalizedMessage() + " " + sm.getString("webappClassLoader.wrongVersion", name));
				}
				entry.loadedClass = clazz;
				entry.binaryContent = null;
				entry.source = null;
				entry.codeBase = null;
				entry.manifest = null;
				entry.certificates = null;
			} else {
				clazz = entry.loadedClass;
			}
		}
		return clazz;
	}

 

Java代码 复制代码 收藏代码
  1. protected ResourceEntry findResourceInternal(String name, String path) {   
  2.     // 先根据类名从缓存中查找对应资源 ,有则直接返回   
  3.     ResourceEntry entry = (ResourceEntry) resourceEntries.get(name);   
  4.     if (entry != null)   
  5.         return entry;   
  6.     int contentLength = -1;   
  7.     // 资源二进制数据长度   
  8.     InputStream binaryStream = null;   
  9.     // 资源二进制输入流   
  10.     int jarFilesLength = jarFiles.length;   
  11.     // classpath中的jar包个数   
  12.     int repositoriesLength = repositories.length;   
  13.     // 仓库数(classpath每一段称为repository仓库)   
  14.     int i;   
  15.     Resource resource = null;   
  16.     // 加载的资源实体   
  17.     boolean fileNeedConvert = false;   
  18.     // 对每个仓库迭代,直接找到相应的entry,如果查找的资源是一个独立的文件,在这个代码块可以查找到相应资源,   
  19.     // 如果是包含在jar包中的类,这段代码并不能找出其对应的资源   
  20.     for (i = 0; (entry == null) && (i < repositoriesLength); i++) {   
  21.         try {   
  22.             String fullPath = repositories[i] + path;   
  23.             // 仓库路径 加资源路径得到全路径   
  24.             Object lookupResult = resources.lookup(fullPath);   
  25.             // 从资源库中查找资源   
  26.             if (lookupResult instanceof Resource) {   
  27.                 resource = (Resource) lookupResult;   
  28.             }   
  29.             // 到这里没有抛出异常,说明资源已经找到,现在构造entry对象   
  30.             if (securityManager != null) {   
  31.                 PrivilegedAction dp = new PrivilegedFindResource(files[i], path);   
  32.                 entry = (ResourceEntry) AccessController.doPrivileged(dp);   
  33.             } else {   
  34.                 entry = findResourceInternal(files[i], path);   
  35.                 // 这个方式只是构造entry并给其codebase和source赋值   
  36.             }   
  37.             // 获取资源长度和最后修改时间   
  38.             ResourceAttributes attributes = (ResourceAttributes) resources.getAttributes(fullPath);   
  39.             contentLength = (int) attributes.getContentLength();   
  40.             entry.lastModified = attributes.getLastModified();   
  41.             // 资源找到,将二进制输入流赋给binaryStream   
  42.             if (resource != null) {   
  43.                 try {   
  44.                     binaryStream = resource.streamContent();   
  45.                 } catch (IOException e) {   
  46.                     return null;   
  47.                 }   
  48.                 // 将资源的最后修改时间加到列表中去,代码略去,参加源码   
  49.             }   
  50.   
  51.         } catch (NamingException e) {   
  52.         }   
  53.     }   
  54.   
  55.     if ((entry == null) && (notFoundResources.containsKey(name)))   
  56.         return null;   
  57.     // 开始从jar包中查找   
  58.     JarEntry jarEntry = null;   
  59.     synchronized (jarFiles) {   
  60.         if (!openJARs()) {   
  61.             return null;   
  62.         }   
  63.         for (i = 0; (entry == null) && (i < jarFilesLength); i++) {   
  64.             jarEntry = jarFiles[i].getJarEntry(path);   
  65.             // 根据路径从jar包中查找资源   
  66.             // 如果jar包中找到资源,则定义entry并将二进制流等数据赋给entry,同时将jar包解压到workdir.   
  67.             if (jarEntry != null) {   
  68.                 entry = new ResourceEntry();   
  69.                 try {   
  70.                     entry.codeBase = getURL(jarRealFiles[i], false);   
  71.                     String jarFakeUrl = getURI(jarRealFiles[i]).toString();   
  72.                     jarFakeUrl = "jar:" + jarFakeUrl + "!/" + path;   
  73.                     entry.source = new URL(jarFakeUrl);   
  74.                     entry.lastModified = jarRealFiles[i].lastModified();   
  75.                 } catch (MalformedURLException e) {   
  76.                     return null;   
  77.                 }   
  78.                 contentLength = (int) jarEntry.getSize();   
  79.                 try {   
  80.                     entry.manifest = jarFiles[i].getManifest();   
  81.                     binaryStream = jarFiles[i].getInputStream(jarEntry);   
  82.                 } catch (IOException e) {   
  83.                     return null;   
  84.                 }   
  85.                 if (antiJARLocking && !(path.endsWith(".class"))) {   
  86.                     // 解压jar包代码,参见源码   
  87.                 }   
  88.   
  89.             }   
  90.   
  91.             if (entry == null) {   
  92.                 synchronized (notFoundResources) {   
  93.                     notFoundResources.put(name, name);   
  94.                 }   
  95.                 return null;   
  96.             }   
  97.             // 从二进制流将资源内容读出   
  98.             if (binaryStream != null) {   
  99.                 byte[] binaryContent = new byte[contentLength];   
  100.                 // 读二进制流的代码,这里省去,参见源码   
  101.                 entry.binaryContent = binaryContent;   
  102.             }   
  103.         }   
  104.         // 将资源加到缓存中   
  105.         synchronized (resourceEntries) {   
  106.             ResourceEntry entry2 = (ResourceEntry) resourceEntries.get(name);   
  107.             if (entry2 == null) {   
  108.                 resourceEntries.put(name, entry);   
  109.             } else {   
  110.                 entry = entry2;   
  111.             }   
  112.         }   
  113.         return entry;   
  114.     }   
  115. }  

 
  • 大小: 42.7 KB
分享到:
评论

相关推荐

    java类加载器-tomcat中的类加载器

    总之,理解Java类加载器和Tomcat的类加载机制对于开发和调试基于Tomcat的应用程序至关重要。它涉及到类的查找、加载、初始化等过程,以及如何通过调整类加载器配置来优化应用性能和安全性。深入研究这些概念,能够让...

    tomcat类加载器

    这个"DevLoader.zip"文件可能包含与Tomcat自定义类加载器相关的资料,特别是名为"DevLoader"的类加载器,这可能是Tomcat为开发者提供的一种特殊加载器。 首先,我们来理解一下类加载器的基本概念。在Java中,类加载...

    类加载器与Tomcat

    首先,Tomcat的类加载器层次结构由Bootstrap类加载器、Common类加载器、Shared类加载器和Web应用程序类加载器组成。Bootstrap类加载器是JVM启动时的第一个加载器,负责加载JDK的核心类库。Tomcat使用Bootstrap类加载...

    JVM、Tomcat、OSGI等类加载器整理文档

    JVM、OSGI(Open Service Gateway Initiative)和Tomcat等容器都涉及到了类加载器的概念,理解它们的工作原理对优化应用性能和解决依赖冲突至关重要。 1. JVM类加载器: - **父类加载器**:在Java中,类加载器之间...

    类加载器(java)

    Java中的类加载器是JVM(Java虚拟机)的核心组件之一,它们负责将.java源代码编译成的.class字节码文件加载到JVM中,从而使得程序能够运行。类加载器不仅涉及到程序的正常执行,还与Java的动态加载、模块化系统以及...

    Java 类在 Tomcat 中是如何加载的(过程分析)

    JVM中的类加载器包括Bootstrap类加载器、Extension类加载器和Application类加载器。 2. **JVM的类加载策略** JVM采用**父类委托机制**来加载类,这意味着当一个类加载器接收到加载请求时,它首先会委托给其父类...

    tomcat 类加载机制 —— ClassLoader

    在Tomcat中,我们可以通过配置`catalina.properties`文件和`server.xml`文件来调整类加载策略,例如设置自定义的类加载顺序或启用共享类加载器。 此外,Tomcat还支持热部署,即在不重启服务器的情况下更新Web应用的...

    jvm虚拟机参数调优,tomcat调优

    类加载机制的优化主要涉及类加载器的使用和类的初始化。例如,通过`-XX:MaxMetaspaceSize`限制元空间的大小,防止因类数量过多导致的溢出。 接下来,Tomcat调优主要包括以下几个方面: 1. **连接器优化**:`...

    深入探讨 Java 类加载器

    在Web容器(如Tomcat)中,每个Web应用都有自己的类加载器,这样可以确保不同应用的类之间隔离,避免冲突。而在OSGi环境下,类加载器被用来实现模块间的隔离,每个模块(Bundle)都拥有自己的类加载器,实现了更细...

    深入探讨 Java 类加载器.pdf

    - **Web容器**:现代Web容器(如Tomcat、Jetty等)广泛使用类加载器来支持多应用共存环境下的隔离性,每个Web应用都有自己的类加载器,这样可以避免不同应用之间的类冲突问题。 - **OSGi**:OSGi是一个Java平台的...

    Tomcat启动时类加载顺序

    Tomcat通过自定义的类加载器实现了特定的类加载顺序,以确保能够正确处理不同来源的类文件,避免类的重复加载和类版本冲突问题。以下是Tomcat启动过程中类加载的具体顺序: #### 三、类加载顺序详述 1. **Bootstrap...

    Tomcat启动顺序

    Bootstrap类加载器完成后,Tomcat会使用系统类加载器(System ClassLoader)加载服务器的全局配置文件,如`server.xml`。这个文件定义了Tomcat的整体架构,包括服务(Service)、连接器(Connector)和引擎(Engine)等组件...

    jvm.zip_jvm_knownobc_tomcat

    7. **类加载器优化**: Tomcat中的类加载机制也可能影响内存使用。理解Web应用程序的类加载层次结构,以及如何配置ClassLoader以避免类冲突和内存浪费,是优化JVM的关键。 综上所述,"jvm.zip_jvm_knownobc_tomcat...

    设置tomcat的jvm虚拟机大小

    JVM 由类加载器、运行时数据区和执行引擎组成。 2. 设置 JVM虚拟机大小的必要性:Tomcat 无法直接在计算机上运行,需要依赖于硬件基础之上的操作系统和一个 Java 虚拟机。因此,需要设置 JVM 虚拟机的大小来满足...

    面试必问之jvm与性能优化.docx

    本文将深入探讨JVM加载Class文件的机制,类加载器的概念及其类型,以及Tomcat服务器特有的类加载机制。 1. JVM加载Class文件的原理机制 JVM加载Class文件的过程分为三个主要阶段:装载、连接和初始化。 - **装载*...

    面试必问之jvm与性能优化.pdf

    在Tomcat服务器中,类的加载机制有所不同,主要包括以下几种类型的类加载器: - **Bootstrap ClassLoader**:加载JVM启动所需的核心类和标准扩展类。 - **System ClassLoader**:加载Tomcat启动所需的类文件,如`...

    Tomcat加载顺序

    这是JVM自带的类加载器,用于加载核心的类库,如`java.lang`、`java.util`等,这些类库存储在`$JAVA_HOME/jre/lib`目录下。Bootstrap ClassLoader不会从磁盘上加载任何类,而是直接读取JRE中的这些核心类库。 #### ...

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

    在Java世界中,类加载器(ClassLoader)是Java运行时环境的重要组成部分,它负责加载类的字节码到JVM中。Tomcat作为一款广泛使用的Java应用服务器,其类加载机制有着独特的设计,特别是对Java的双亲委派模型有所突破...

    2020年携程-Java高级.pdf

    Java的类加载器主要有三种:Bootstrap类加载器、Extension类加载器和Application类加载器。Bootstrap加载器负责加载JDK的核心类库,如rt.jar,它是所有其他类加载器的父加载器,但本身没有父加载器。Extension类加载...

Global site tag (gtag.js) - Google Analytics