7.JVM基础-虚拟机类加载机制(二)

7.3 类加载器

7.3.1 类与类加载器

对于任意一个类,都需要由加载它的类加载器和这个类本身一同确立其在 java 虚拟机中的唯一性,每一类加载器,都拥有一个类加载器,都拥有一个独立的类名称空间。简单来说,就是“比较两个类是都相等”。只有在这两个类是由同一个加载器加载的提前下才有意义。否则,即使这两个类来源同一个 Class 文件,被同一个虚拟机加载,只要记在它们的类加载器不同,那这两个类就必定不相等。

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

package com.liys.test;

import java.io.IOException;
import java.io.InputStream;

public class ClassLoaderTest {

    public static void main(String[] args) throws Exception {

        ClassLoader myClassLoader = new ClassLoader() {
            @Override
            public Class<?> loadClass(String name) throws ClassNotFoundException {
                try {
                    String fileName = name.substring(name.lastIndexOf(".") + 1) + ".class";
                    InputStream in  = getClass().getResourceAsStream(fileName);

                    if (in == null) {
                        return super.loadClass(name);
                    }

                    byte[] b = new byte[in.available()];
                    in.read(b);
                    return defineClass(name, b, 0, b.length);
                } catch (IOException e) {
                    throw new ClassNotFoundException(name);
                }
            }
        };

        Object obj = myClassLoader.loadClass("com.liys.test.ClassLoaderTest").newInstance();
        System.out.println(obj.getClass());
        System.out.println(obj instanceof com.liys.test.ClassLoaderTest);
    }
}
class com.liys.test.ClassLoaderTest
false

这是因为虚拟机中存在了两个 ClassLoaderTest 类,一个是由系统应用程序类加载器加载的,另一个由我们自己定义的类加载器加载的,虽然都是同一个 Class 文件,但是依然是两个独立的类,因此做对象所属类型检查的结果自然为 false。

 

7.3.2 双亲委派模型

从 java 虚拟机的角度来说,只存在两种不同的类加载器:一种是启动类加载器,这类加载器使用c++实现,是虚拟机的一部分。另一种就是所有其他的类加载器,这些类加载器都由 java 实现,独立于虚拟机外部,并且全部都继承自抽象类 java.lang.ClassLoader。绝大部分的 java 程序都会使用到一下3种系统提供的类加载器:

1.  启动类加载器(Bootstrap ClassLoader):前面已经介绍过,这个类加载器负责将存放在 JAVA_HOME\lib 目录中,或者被 -Xbootclasspath 参数所设定的路径中的,并且是虚拟机识别的类库加载到虚拟机内存中。

启动类加载器无法被 java 程序直接引用,用户在编写自定义类加载器时,如果需要把请求委派给引导类加载,直接使用null即可。

2. 扩展类加载器(Extension ClassLoader):这个加载器由 sun.misc.Launcher$ExtClassLoader 实现,它负责加载 JAVA_HOME\lib\ext 目录中的,或者被 java.ext.dirs 系统变量所指定的路径中的所有类库,开发者可以直接使用扩展类加载器。

3. 应用程序类加载器(Application ClassLoader):这类加载器由 sun.misc.Launcher$ApplicationClassLoader 实现。由于这个类加载器是 ClassLoader 中的 getSystemClassLoader() 方法的返回值,所以一般也称它为系统类加载器。它负责用户类路径上所指定的类库,开发者可以直接使用这个类的加载器,如果应用程序中没有自定义过自己的类加载器,一般情况下这个就是程序中默认的类加载程序。

 

 

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

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

 

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值