美文网首页Android开发Android基础相关
类加载器、双亲委派模型

类加载器、双亲委派模型

作者: Android小马哥 | 来源:发表于2018-08-02 10:35 被阅读2次

    说明:其实参考的文章已经很详细了。想看一个知识点的时候,会去看好几篇文章。进行一下总结。有时候一篇文章就写得很好,但是,不自己写一下的话,总感觉不够,所以就用符合自己理解或者记忆的方式写一下。因为写的时候,相当于心里说了几遍。面试的时候就能表达出来吧。所以简洁的写了下,参考的文章附在后面。

    类加载器分类

    启动(Bootstrap)类加载器
    • 负责把存放在<JAVA_HOME>\lib目录下或被-Xbootclasspath参数指定的路径中,
      能被虚拟机识别(仅按文件名识别rt.jar,名字不符合的类库不会被加载)的类库加载到虚拟机内存中。
    • 该类加载器无法被应用程序直接引用,自定义类加载器时如果要把加载请求委派给它加载,直接用null代替即可
    扩展类加载器(Extension ClassLoader)
    • 该加载器由sum.misc.Launcher$ExtClassLoader实现,负责加载<JAVA_HOME>\lib\ext目录中或者java.ext.dirs系统变量指定的路径中所有类库。
    应用程序类加载器(Application ClassLoader)/也称为系统类加载器
    • 该加载器由sum.misc.Launcher$App-ClassLoader实现,在ClassLoader类中的getSystemClassLoader()方法返回该类加载器。负责加载用户类路径(ClassPath)上所指定的类库,程序默认使用这个类加载器。
    自定义类加载器

    类加载器与类的唯一性

    类加载器虽然只用于实现类的加载动作,但是对于任意一个类,都需要由加载它的类加载器和这个类本身共同确立其在Java虚拟机中的唯一性。通俗的说,JVM中两个类是否“相等”,首先就必须是同一个类加载器加载的,否则,即使这两个类来源于同一个Class文件,被同一个虚拟机加载,只要类加载器不同,那么这两个类必定是不相等的。

    这里的“相等”,包括代表类的Class对象的equals()方法、isAssignableFrom()方法、isInstance()方法的返回结果,也包括使用instanceof关键字做对象所属关系判定等情况。

    双亲委派模型

    protected Class<?> loadClass(String name, boolean resolve) throws ClassNotFoundException {
        synchronized (getClassLoadingLock(name)) {
        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) {
        }
        if (c == null) { //递归调用已让所有的父类尝试过,还为null的话
            long t1 = System.nanoTime();
            c = findClass(name);//父类加载器处理不了,就由加载器的findClass方法来加载
            sun.misc.PerfCounter.getParentDelegationTime().addTime(t1 - t0);
            sun.misc.PerfCounter.getFindClassTime().addElapsedTimeFrom(t1);
            sun.misc.PerfCounter.getFindClasses().increment();
        }
        }
        if (resolve) {
        resolveClass(c);
        }
        return c;
    }
                }
    
    
    好处

    保证了唯一性
    使用双亲委派模型的好处在于Java类随着它的类加载器一起具备了一种带有优先级的层次关系。例如类java.lang.Object,它存在在rt.jar中,无论哪一个类加载器要加载这个类,最终都是委派给处于模型最顶端的Bootstrap ClassLoader进行加载,因此Object类在程序的各种类加载器环境中都是同一个类。相反,如果没有双亲委派模型而是由各个类加载器自行加载的话,如果用户编写了一个java.lang.Object的同名类并放在ClassPath中,那系统中将会出现多个不同的Object类,程序将混乱。因此,如果开发者尝试编写一个与rt.jar类库中重名的Java类,可以正常编译,但是永远无法被加载运行。

    而且 从双亲委派模型的实现可知,如果自定义类加载器,覆盖了loadClass方法,会导致双亲委派模型被破坏,那么该自定义类加载器与启动类加载器加载同一个class得到的也不会是相同的类对象。 所以自定义类加载器,正常不要去覆盖loadClass方法,但是可以重新findClass方法。

    参考 、感谢

    文章中的文字来自于下面两篇文章。比较详细。
    【深入理解JVM】:类加载器与双亲委派模型
    jvm学习笔记(四)虚拟机类加载机制、类加载器

    相关文章

      网友评论

        本文标题:类加载器、双亲委派模型

        本文链接:https://www.haomeiwen.com/subject/qecjvftx.html