JVM类加载流程及双亲委派机制

一、类加载流程

JVM类加载流程及双亲委派机制_第1张图片
JVM类加载流程及双亲委派机制_第2张图片

二、类加载器

JVM类加载流程及双亲委派机制_第3张图片
证明理论成立的代码:

public class TestMain {
     

    //
    public static void main(String[] args) {
     
        System.out.println(TestMain.class.getClassLoader());
        System.out.println(TestMain.class.getClassLoader().getParent());
        System.out.println(TestMain.class.getClassLoader().getParent().getParent());

    }
}

执行结果:
JVM类加载流程及双亲委派机制_第4张图片
因为bootstrap classLoader是C语言实现的,无法在java代码中捕捉到,因此这里打印的是null,其余两个类加载器都已经打印出来了。

三、源码分析

查看java.lang.ClassLoader的loadClass(java.lang.String, boolean)方法的源码:

protected Class<?> loadClass(String name, boolean resolve)
            throws ClassNotFoundException
    {
     
        synchronized (getClassLoadingLock(name)) {
     
            // 首先检查这个classsh是否已经加载过了
            Class<?> c = findLoadedClass(name);
            if (c == null) {
     
                long t0 = System.nanoTime();
                try {
     
                    // c==null表示没有加载,如果有父类的加载器则让父类加载器加载
                    if (parent != null) {
     
                        c = parent.loadClass(name, false);
                    } else {
     
                        //如果父类的加载器为空 则说明递归到bootStrapClassloader了
                        //bootStrapClassloader比较特殊无法通过get获取
                        c = findBootstrapClassOrNull(name);
                    }
                } catch (ClassNotFoundException e) {
     }
                if (c == null) {
     
                    //如果bootstrapClassLoader 仍然没有加载过,则递归回来,尝试自己去加载class
                    long t1 = System.nanoTime();
                    c = findClass(name);
                    sun.misc.PerfCounter.getParentDelegationTime().addTime(t1 - t0);
                    sun.misc.PerfCounter.getFindClassTime().addElapsedTimeFrom(t1);
                    sun.misc.PerfCounter.getFindClasses().increment();
                }
            }
            if (resolve) {
     
                resolveClass(c);
            }
            return c;
        }
    }

四、双亲委派机制

当某个类加载器需要加载某个.class文件时,它首先把这个任务委托给他的上级类加载器,递归这个操作,如果上级的类加载器没有加载,自己才会去加载这个类,流程如下:
JVM类加载流程及双亲委派机制_第5张图片
双亲委派机制的作用:
1、防止重复加载同一个.class。通过委托去向上面问一问,加载过了,就不用再加载一遍。保证数据安全。
2、保证核心.class不能被篡改。通过委托方式,不会去篡改核心.clas,即使篡改也不会去加载,即使加载也不会是同一个.class对象了。不同的加载器加载同一个.class也不是同一个Class对象。这样保证了Class执行安全。

你可能感兴趣的:(java基础,java)