JVM---双亲委派模型(源码详解)

含义

1.双亲委派模型是从JDK1.2开始至今都在沿用的类加载模型,具体的逻辑是:
先看源码:

 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;
        }
    }

注释已经很清晰了:
首先,根据类的全限定名通过findLoadedClass方法查询该类是否已被加载,如果已经被加载过了则判断resolve的值,该值是标识类是否link了,(This (misleadingly named) method may be
used by a class loader to link a class.)默认传false,也就是说直接返回了该类.重点是如果该类没有加载过:再判断是否有父类加载器(通过组合的方式持有),如果有父类加载器,就用父类加载器加载,如果没有说明其上层只剩启动类加载器了,就用启动类加载器加载,(因为启动类加载器是通过c++写的).

如果其父类加载器或者启动类加载器都不能加载,会抛出异常,然后由本类加载器的findClass方法进行加载,然后记录类加载器的的状态信息

优点

优点也就是使用该模型的必要性:这种加载机制可以使类连同其加载器有了一种具有优先级的层次关系,保证了同一个类的全限定名只对应一个类,保证了其唯一性,否者,根本就没法保证java的工作机制.

缺点

显而易见,如果上层的类加载器在工作的时候需要下层的类加载器进行加载操作则是不支持的,但这种场景有很多的场景,比如,JNDI,JDBC,JBI等涉及到SPI(接口提供者)
的地方,都是为了解决这个问题而提出的方式,具体可以看下篇 破坏双亲委派模型.这里不做细致的记录.

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章