ClassLoader学习

JVM 类加载器常用的三种方式

1. 默认方式 之双亲委托模式

ClassLoader学习_第1张图片
JVM默认的三种加载器

子类加载器在加载一个类时,会让父类先加载。如果父类已经加载了的时候,没有必要让子类再加载一次。

protected Class loadClass(String name, boolean resolve)
        throws ClassNotFoundException
    {
        synchronized (getClassLoadingLock(name)) {
            // 首先,检查从JVM缓存查找该类
            Class c = findLoadedClass(name);
            if (c == null) {
                long t0 = System.nanoTime();
                try {
                    if (parent != null) { // 然后委托给父类加载器进行加载
                        c = parent.loadClass(name, false);
                    } else { // 如果父类加载器为null, 则委托给Bootstrap类加载器加载
                        c = findBootstrapClassOrNull(name);
                    }
                } catch (ClassNotFoundException e) {
                    // ClassNotFoundException thrown if class not found
                    // from the non-null parent class loader
                }

                if (c == null) {
                    // 如果仍然没有找到,则调用findClass()进行查找 
                    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;
        }
    }

从以上代码可知,实现自己的类加载策略可以覆盖ClassLoader的findClass() 或者loadClass()方法来实现。

2. 父类加载器加载子类加载器中的资源

如果一个类引用了另一个类,那么被引用的类应该有引用方类加载器加载。

通过设置和获取线程上下文类加载器ContextClassLoader实现

    ClassLoader extClassLoader =   MyContextClassLoad.class.getClassLoader().getParent();
    Thread.currentThread().setContextClassLoader(extClassLoader);

3. 类加载器要使用不在当前类加载器查找路径的类

新建一个在指定路径查找类的类加载器

你可能感兴趣的:(ClassLoader学习)