JVM篇-类加载器

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

类与类加载器

任意一个类。都需要由加载它的类加载器和这个类本身一同确立其在java虚拟机的唯一性,每一个类加载器都拥有一个独立的类名称空间。即使两个类来源于同一个class文件,被同一个虚拟机加载,只要加载他们的类加载器不同,那这两个类就必定不相等。

不同类加载器对instanceof的影响

类加载器

  • 启动类加载器(Bootstrap ClassLoader):使用C++语言实现(大名鼎鼎的HotSpot)。将存放在<JAVA_HOME>\lib目录中的,或者被-Xbootclasspath参数所指定的路径中的,并且是虚拟机识别的类库(仅按照文件名识别,名字不符合的类库即使放在lib目录中也不会被加载)加载到虚拟机内存中。
  • 扩展类加载器(Extension ClassLoader):由sun.misc.Launcher$ExtClassLoader实现,他负责加载<JAVA_HOME>\lib\ext目录中的,或者被java.ext.dirs系统变量所指定的路径中所有的类库,开发者可直接使用。
  • 应用程序类加载器(Application ClassLoader):由sun.misc.Launcher$AppClassLoader实现,也叫系统类加载器。负责加载用户类路径(classpath)上所指定的类库,开发者可直接使用。

类加载器之间的层次关系

双亲委派模型

如上图所示的模型被称之为“双亲委派模型”。

双亲委派模型工作流程:

如果一个类加载器收到了类加载的请求,它首先会把这个请求委派给父类加载器,每一层类加载器都是如此(注意:启动类加载器没有父类,若委派到启动类加载器时还没有找到这个类,会抛弃异常:ClassNotFoundException)。因此所有的加载请求最终都应该委派到最顶层的启动类加载器中,只有当父加载器反馈自己无法完成这个加载请求时,子加载器才会尝试自己去完成加载。

protected Class<?> loadClass(String name, boolean resolve)
        throws ClassNotFoundException
    {
        synchronized (getClassLoadingLock(name)) {
            // First, check if the class has already been loaded
            Class<?> c = findLoadedClass(name);
            if (c == null) {
                long t0 = System.nanoTime();
                try {
                    if (parent != null) {
                        c = parent.loadClass(name, false);
                    } else {
                        c = findBootstrapClassOrNull(name);
                    }
                } catch (ClassNotFoundException e) {
                    // ClassNotFoundException thrown if class not found
                    // from the non-null parent class loader
                }

                if (c == null) {
                    // If still not found, then invoke findClass in order
                    // to find the class.
                    long t1 = System.nanoTime();
                    c = findClass(name);

                    // this is the defining class loader; record the stats
                    sun.misc.PerfCounter.getParentDelegationTime().addTime(t1 - t0);
                    sun.misc.PerfCounter.getFindClassTime().addElapsedTimeFrom(t1);
                    sun.misc.PerfCounter.getFindClasses().increment();
                }
            }
            if (resolve) {
                resolveClass(c);
            }
            return c;
        }
    }

JDK类加载源码解读:
先检查请求加载的类是否已经被加载过,若没有被加载过则调用父类加载器的loadClass()方法,若父类加载器为空则默认使用启动类加载器作为父加载器。若父加载器加载失败,抛出ClassNotFoundException,此时才调用自己的findClass()方法尝试加载。加载完成后若加载类没有被解析过,则进行解析。

打破双亲委派

双亲委派有一个很大的局限性就是无法做到不委派向下委派

java.sql.Driver的实现需要各个数据库的服务商(MySQL/Oracle…)来提供,由于Driver接口定义在jdk当中的,DriverManager(也由jdk提供)要加载各个实现了Driver接口的实现类,然后进行管理,但是DriverManager由启动类加载器加载,只能记载JAVA_HOME的lib下文件,而它的实现是由服务商提供的,此时就需要启动类加载器来委托子类来加载Driver实现,从而打破双亲委派。这就是常说的SPI(Service Provider Interface)机制。

向下委派

通过java.util.ServiceLoader可以实现向下委派。

public static <S> ServiceLoader<S> load(Class<S> service) {
        ClassLoader cl = Thread.currentThread().getContextClassLoader();
        return ServiceLoader.load(service, cl);
    }

通过ServiceLoader的源码可以看出,ServiceLoader的核心就是线程上下文类加载器(ContextClassLoader)。

SPI机制实例:JNDI、JDBC、JCE、JAXB、JBI、SpringBoot服务发现等。

沙箱安全机制

防止源码被篡改,若自定义java.lang.String类,类加载时会抛出异常。
源码体现:AccessController.doPrivileged()

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值