JVM(2)--类加载机制--双亲委派原理

前面介绍了类的加载子系统,从一个.class文件变成Java虚拟机可用的类,要经过加载–>链接(验证–>准备–>解析)–>初始化–>使用–>卸载这五个过程,经过这五个过程.class文件变成了可用的类,但是具体到组件,到底是谁执行了上面这五个过程呢?是谁保证这个加载过程中的安全性和准确性?这里就不得不提到类加载器和双亲委派模型

一、类加载器ClassLoader

在虚拟机的角度,一共有两种类加载器,①启动类加载器,②其他的加载器,这样划分的依据是,启动类加载器是使用C/C++编写的,其他加载器是java语言编写的。但是这样划分过于笼统。

站在Java开发人员的角度,类加载器ClassLoader分为两类:①虚拟机自带的加载器,②用户自定义类加载器

虚拟机自带的加载器

1.BootStrap ClassLoader

BootStrap ClassLoader启动类加载器,启动类加载器具有以下特点:

  • 这个类加载使用C/C++语言实现的,嵌套在JVM内部

  • 它用来加载java的核心库(JAVA_HOME/jre/lib/rt.jar/resources.jar或sun.boot.class.path路径下的内容),用于提供JVM自身需要的类

  • 并不继承自java.lang.ClassLoader,没有父加载器

  • 加载拓展类和应用程序类加载器,并指定为他们的父加载器

  • 处于安全考虑,BootStrap启动类加载器只加载包名为java、javax、sun等开头的类

2. Extension ClassLoader

Extension ClassLoader 拓展类加载器

  • java语言编写 ,由sun.misc.Launcher$ExtClassLoader实现。

  • 派生于ClassLoader类

  • 父类加载器为启动类加载器

  • 从java.ext.dirs系统属性所指定的目录中加载类库,或从JDK的安装目录的jre/lib/ext子目录(扩展目录)下加载类库。如果用户创建的JAR放在此目录下,也会由拓展类加载器自动加载

3.AppClassLoader

AppClassLoader应用程序类加载器

  • java语言编写, 由sun.misc.Launcher$AppClassLoader实现。
  • 派生于ClassLoader类
  • 父类加载器为拓展类加载器
  • 它负责加载环境变量classpath或系统属性 java.class.path指定路径下的类库

该类加载器是程序中默认的类加载器,一般来说,java应用的类都是由它来完成加载

通过ClassLoader#getSystemClassLoader()方法可以获取到该类加载器

/**
 * @author 四五又十
 * @version 1.0
 * @date 2020/7/4 9:46
 */
public class demo2 {
    
    
    public static void main(String[] args) {
    
    

        //应用程序加载器  sun.misc.Launcher$AppClassLoader@18b4aac2
        ClassLoader systemClassLoader = ClassLoader.getSystemClassLoader();
        System.out.println(systemClassLoader);

        //扩展类加载器  sun.misc.Launcher$ExtClassLoader@1540e19d
        ClassLoader exClassLoader = systemClassLoader.getParent();
        System.out.println(exClassLoader);

        //试图获取引导类加载器  null
        ClassLoader bootstrapClassLoader = exClassLoader.getParent();
        System.out.println(bootstrapClassLoader);
    }
}

通过上面程序可以发现,引导类加载器Bootstrap ClassLoader获取下来的值是null,这样好理解,因为Bootstrap ClassLoader是使用C/C++语言编写的,自然获取不到。从上面程序也可以发现,Bootstrap ClassLoader、Extension ClassLoader、AppClassLoader存在层级关系

上面分别介绍了这3中加载器的特点,其实这3中加载器的最主要的区别在于,分别加载不同的类:

AppClassLoader:负责加载用户自己写的类

Extension ClassLoader:负责加载<Java_Home>\lib\ext目录下的类

BootStrap ClassLoader:负责加载Java核心类库,BootStrap启动类加载器只加载包名为java、javax、sun等开头的类

例如:尝试获取加载String类的类加载器:String类是由BootStrap ClassLoader加载,那么输出为null

//输出 null
ClassLoader classLoader = String.class.getClassLoader();
System.out.println(classLoader);

获取ClassLoader的方法:

1.获取当前类的ClassLoader: clazz.getClassLoader

2.获取当前线程的ClassLoader:Thread.currentThread().getContextCladdLoader()

3.获取系统的classLoader:ClassLoader.getSystemClassLoader()

4.获取调用者的classLoader:DriverManager.getCallerClassLoader()

4.用户自定义加载器

  • 隔离加载类
  • 修改类加载的方式
  • 拓展加载源
  • 防止源码泄漏

二、双亲委派机制

通过一个实例来恰到好处的说明双亲委派机制的好处,例如:在工作目录下创建一个类,类名和包名为:java.lang.String,这里不需要质疑,有人会问,String类不是Java自带的类吗?这样创建是合法的,编译器并不会报错。在String类中,我们只是去输出一个语句例如:

package java.lang;

/**
 * @author 四五又十
 * @version 1.0
 * @date 2020/7/6 18:29
 */
public class String {
    
    
    static {
    
    
        System.out.println("自定义String类");
    }
}

然后在一个main方法里去new 一个String

public class demo4 {
    
    
    public static void main(String[] args) {
    
    
        String s = new String();
    }
}

现在思考,String类的加载是会加载jdk中的String,还是我们自定义的String方法,毕竟它们都在java.lang包下?如果是加载jdk下的String类,那么控制台不输出,如果是加载自定义中的String,因为在自定义的String类中使用了静态代码块,那么控制台肯定要输出一句话“自定义String类”?

答案是控制台不输出!!也就是Java虚拟机选择加载了jdk中的String,这就是双亲委派模型在起作用

双亲委派机制的工作过程是:如果一个类加载器收到了加载类的请求,那么它不会自己尝试去加载这个类,而是把请求都交给父类加载器,父类加载器无法加载在交给更上一层的父类加载器,依次递归,只有当父类加载器无法加载该类时,子加载器才会尝试加载。

根据双亲委派模型,可以得出加载如上demo4的过程:类加载器收到请求要加载demo4,那么虚拟机会将该类交给父加载器Extension ClassLoaderExtension ClassLoader一看这个类我也没有办法加载,那么再交给父类加载器BootStrap ClassLoaderBootStrap ClassLoader也没法加载,那么还是向下委派,直到AppClassLoaderAppClassLoader是负责加载用户自定义类的,它可以加载,那么demo4就被加载到虚拟机中了。

再看String类的加载过程,首先AppClassLoader收到要加载String类的请求,AppClassLoader自己不会去尝试加载,要交给父类Extension ClassLoaderExtension ClassLoader一看我也没有权限加载,那么在向上委派交给BootStrap ClassLoader,BootStrap ClassLoader一看String是java包下的,加载这种类是属于BootStrap ClassLoader的工作,那么Stirng类就被加载至虚拟机上了,那么既然String类是由BootStrap ClassLoader加载的,那么显然是jdk中的String

在源码中,实现双亲委派机制的代码在java.lang.ClassLoader的loadClass方法中:

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

2.1 双亲委派模型的好处

  • 双亲委派模型规定的这种具有优先级的层次关系,可以避免类的重复加载。当父加载器已经加载过此类时,如果再有加载这种类的请求,那么应用程序类加载器会将这个1请求交到父类加载器,接着BootStrap启动类加载器执行了findLoadedClass方法,发现该类已经被加载过了,则直接返回该类不在进行加载,这就避免了类的重复加载
  • 双亲委派模型的另外一个好处是,避免了核心类被修改,通过上面的例子便可以知道,一般核心类是指rt.jar或者tools.jar包下的一些类,如果这张类被恶意修改那么将会出现意向不到的后果,而这种双亲委派模型加载类的层次结构保证了核心类被加载,因为启动类只加载正确路径下的类

2.2 打破双亲委派模型

双亲委派模型保证了各个类加载器的基础类一致的问题,就是这些基础类都交给上层的类加载器进行加载,但是这里也会带来一个问题。在Java应用中存在很多服务核心调用接口(简称SPI),这些接口允许第三方为它实现,例如JDBC等等,这些SPI的接口属于核心类库,则应该交给BootStrap启动类加载器进行加载,但是问题出现在这些接口的实现类是第三方接口,从上面已经知道了第三方的库应该交给应用程序加载器AppClassLoader进行加载,由于双亲委派的存在,Bootstrap无法委派AppClassLoader加载这些SPI的实现类

应用中存在很多服务核心调用接口(简称SPI),这些接口允许第三方为它实现,例如JDBC等等,这些SPI的接口属于核心类库,则应该交给BootStrap启动类加载器进行加载,但是问题出现在这些接口的实现类是第三方接口,从上面已经知道了第三方的库应该交给应用程序加载器AppClassLoader进行加载,由于双亲委派的存在,Bootstrap无法委派AppClassLoader加载这些SPI的实现类

要解决这个问题,就需要引入一些特殊的第三方加载器,而线程上下文类加载器就是这样一种打破了双亲委派机制的类加载器,引入了线程上下文加载器,程序可以把本来应该由启动类加载器加载的类交由自定义加载器进行加载,实现了父类加载器委托子类加载器加载类,这样就打通了双亲委派机制下的父类无法委托子类进行加载的问题

猜你喜欢

转载自blog.csdn.net/weixin_44706647/article/details/115140343