1. 编译和类加载
首先说一下类加载机制。
我们自己写的.java文件到最终运行,必须经过编译和类加载这两个阶段。
编译的过程就是把.java文件编译成.class文件。
类加载就是把.class文件加载到JVM内存里面。装载完成以后会得到一个class对象,就可以使用new关键字来实例化这个对象。
![](https://i-blog.csdnimg.cn/direct/f8cbcf8b1718430f99263247f7f6a58a.png)
类加载过程需要涉及到类加载器,JVM在运行的时候会产生三个类加载器,这三个类加载器组成了一个层级关系,每一个类加载器,分别去加载不同作用范围的jar包。
启动类加载器:主要负责Java核心类库的加载,也就是 {JRE_HOME}/lib/rt.jar 和 {JRE_HOME}/lib/resources.jar
扩展类加载器: {JRE_HOME}/lib/ext/*.jar和class文件
应用类加载器:主要负责当前应用里面classpath下面的所有jar包和class文件
自定义类加载器:除了系统提供的类加载器,还可以通过ClassLoader类来实现自定义加载器,去满足一些特殊场景的需求。
![](https://i-blog.csdnimg.cn/direct/810be7412beb4f70ba2a58be6c82f26a.png)
2. 双亲委派模型
双亲委派模型就是按照类加载器的层级关系,逐层进行委派。
比如,当加载一个.class文件的时候, 首先会去把这个.class文件的查询和加载委派给父加载器去执行,如果父加载器都无法加载,那么再尝试自己来加载这样一个.class文件。
好处有两个:
安全性。这种层级关系实际上代表的是一种优先级,也就是说所有的类加载优先要给到启动类加载器,对于核心类库中的一些类,就没有办法被破坏。比如自己写了一个java.lang.String,最终还是会交给启动类加载器。再加上每个类加载器的本身的一个作用范围,那么自己写的java.lang.String就没有办法去覆盖类库中的类。
层级关系的设计可以避免重复加载导致程序混乱的问题。因为父加载器已经加载过了,子加载器就没有必要再去加载了。
![](https://i-blog.csdnimg.cn/direct/ed72751e07d74c328795819792489525.png)
![](https://i-blog.csdnimg.cn/direct/a44d777a502244daa469db804a5a3fea.png)