今日鸡汤:不学原理或许可以走的很快,但是学了原理会帮我们走的更远 …
我们点击运行,然后我们的 Java 代码就会被编译器编译成 .class 字节码文件,然后我们的 .class 字节码文件就会被交给 JVM 进行类加载 。下面是 JVM 根据得来的 .class 字节码文件 创建一个对象的对象创建流程如下:

在一个对象创建流程中,我们的类加载器会判断我们的这个类是否已经被加载,加载好的类具有类的所有数据访问入口,这样才能被分配内存,进行后续的操作,如果说类没被加载,我们JVM就会帮我们去加载这个类,我们的双亲委派机制就发生在这个过程中 。
JVM 具有 4 种类加载器:
代码示例:
import sun.misc.Launcher;
import java.net.URL;
/**
* @Author: WanqingLiu
* @Date: 2022/12/02/15:11
*/
public class DemoClassLoader {
public static void main(String[] args) {
// String 是最高级的类加载器,也就是我们的引导类加载器加载器
// 打印 null ———— Java 中无法打印出引导类加载器地址
System.out.println(String.class.getClassLoader());
// 扩展类加载器加载的
// sun.misc.Launcher$ExtClassLoader@7ea987ac
System.out.println(com.sun.crypto.provider.DESKeyFactory.class.getClassLoader());
// 我们自己写的类肯定是应用类加载的
// sun.misc.Launcher$AppClassLoader@18b4aac2
System.out.println(TestJDKClassLoader.class.getClassLoader());
// 类加载器具有父子级关系
ClassLoader appClassLoader = ClassLoader.getSystemClassLoader();
ClassLoader extClassLoader = appClassLoader.getParent();
ClassLoader bootstrapClassLoader = extClassLoader.getParent();
System.out.println("引导类加载器为:" + bootstrapClassLoader);
System.out.println("扩展类加载为:" + extClassLoader);
System.out.println("应用类加载器为:" + appClassLoader);
System.out.println("***引导类加载器加载的文件有:***");
URL[] urls = Launcher.getBootstrapClassPath().getURLs();
for(URL url : urls){
System.out.println(url);
}
System.out.println("***扩展类加载器加载的文件有:***");
String extClassLoaderFiles = System.getProperty("java.ext.dirs");
String[] extClassLoaderFilesArr = extClassLoaderFiles.split(";");
for (String str : extClassLoaderFilesArr){
System.out.println(str);
}
System.out.println("***应用类加载器加载的文件有:***");
String appClassLoaderFiles = System.getProperty("java.class.path");
String[] appClassLoaderFilesArr = appClassLoaderFiles.split(";");
for (String str : appClassLoaderFilesArr){
System.out.println(str);
}
}
}
结果验证了我们上文提到的关于不同类加载器加载的内容 ——
***引导类加载器加载的文件有:***
file:/C:/Program%20Files/Java/jdk1.8.0_181/jre/lib/resources.jar
file:/C:/Program%20Files/Java/jdk1.8.0_181/jre/lib/rt.jar
file:/C:/Program%20Files/Java/jdk1.8.0_181/jre/lib/sunrsasign.jar
file:/C:/Program%20Files/Java/jdk1.8.0_181/jre/lib/jsse.jar
file:/C:/Program%20Files/Java/jdk1.8.0_181/jre/lib/jce.jar
file:/C:/Program%20Files/Java/jdk1.8.0_181/jre/lib/charsets.jar
file:/C:/Program%20Files/Java/jdk1.8.0_181/jre/lib/jfr.jar
file:/C:/Program%20Files/Java/jdk1.8.0_181/jre/classes
***扩展类加载器加载的文件有:***
C:\Program Files\Java\jdk1.8.0_181\jre\lib\ext
C:\Windows\Sun\Java\lib\ext
***应用类加载器加载的文件有:***
C:\Program Files\Java\jdk1.8.0_181\jre\lib\charsets.jar
C:\Program Files\Java\jdk1.8.0_181\jre\lib\deploy.jar
C:\Program Files\Java\jdk1.8.0_181\jre\lib\ext\access-bridge-64.jar
C:\Program Files\Java\jdk1.8.0_181\jre\lib\ext\cldrdata.jar
C:\Program Files\Java\jdk1.8.0_181\jre\lib\ext\dnsns.jar
C:\Program Files\Java\jdk1.8.0_181\jre\lib\ext\jaccess.jar
C:\Program Files\Java\jdk1.8.0_181\jre\lib\ext\jfxrt.jar
C:\Program Files\Java\jdk1.8.0_181\jre\lib\ext\localedata.jar
C:\Program Files\Java\jdk1.8.0_181\jre\lib\ext\nashorn.jar
C:\Program Files\Java\jdk1.8.0_181\jre\lib\ext\sunec.jar
C:\Program Files\Java\jdk1.8.0_181\jre\lib\ext\sunjce_provider.jar
C:\Program Files\Java\jdk1.8.0_181\jre\lib\ext\sunmscapi.jar
C:\Program Files\Java\jdk1.8.0_181\jre\lib\ext\sunpkcs11.jar
C:\Program Files\Java\jdk1.8.0_181\jre\lib\ext\zipfs.jar
C:\Program Files\Java\jdk1.8.0_181\jre\lib\javaws.jar
C:\Program Files\Java\jdk1.8.0_181\jre\lib\jce.jar
C:\Program Files\Java\jdk1.8.0_181\jre\lib\jfr.jar
C:\Program Files\Java\jdk1.8.0_181\jre\lib\jfxswt.jar
C:\Program Files\Java\jdk1.8.0_181\jre\lib\jsse.jar
C:\Program Files\Java\jdk1.8.0_181\jre\lib\management-agent.jar
C:\Program Files\Java\jdk1.8.0_181\jre\lib\plugin.jar
C:\Program Files\Java\jdk1.8.0_181\jre\lib\resources.jar
C:\Program Files\Java\jdk1.8.0_181\jre\lib\rt.jar
C:\Users\15141\Desktop\headCount\out\production\headCount
D:\IntelliJ IDEA 2022.2.3\lib\idea_rt.jar
Process finished with exit code 0
先向上委托找到父亲加载器加载,父加载器加载不了再下退请求再由儿子加载器自己加载,其流程如下:

好处:
Tomcat 打破了双亲委派机制,原因如下:
不同应用程序依赖的 war 包使用了相同名称的类,但是这些类版本不同。如果使用双亲委派机制,只能加载一个,不能把不同版本的都加载进去。
再详细解释:
我们都用过 Tomcat 服务器,一个 Tomcat 服务器上可能同时部署多个应用程序,这多个应用程序可能使用的第三方类库相同,但是版本不同,也就是说存在类名相同,但是内容不同的类。这时,假如我们使用了双亲委派机制,那就只能加载一个进去。很显然,这是不合理的,因此 Tomcat 使用自定义类加载器,而不使用双亲委派机制。
Tomcat 通过自定义类加载器的方式解决上述问题,其示意图如下:

