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) {System.out.println(String.class.getClassLoader());System.out.println(com.sun.crypto.provider.DESKeyFactory.class.getClassLoader());System.out.println(TestJDKClassLoader.class.getClassLoader());System.out.println();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();System.out.println("***引导类加载器加载的文件有:***");URL[] urls = Launcher.getBootstrapClassPath().getURLs();for(URL url : urls){System.out.println(url);}System.out.println();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();System.out.println("***应用类加载器加载的文件有:***");String appClassLoaderFiles = System.getProperty("java.class.path");String[] appClassLoaderFilesArr = appClassLoaderFiles.split(";");for (String str : appClassLoaderFilesArr){System.out.println(str);}}
}
结果输出:
null
sun.misc.Launcher$ExtClassLoader@7ea987ac
sun.misc.Launcher$AppClassLoader@18b4aac2引导类加载器为:null
扩展类加载为:sun.misc.Launcher$ExtClassLoader@7ea987ac
应用类加载器为:sun.misc.Launcher$AppClassLoader@18b4aac2***引导类加载器加载的文件有:***
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.jarProcess finished with exit code 0
先找父亲加载器加载,不行再由儿子加载器自己加载,其流程如下:
好处:
Tomcat 打破了双亲委派机制,原因如下:
不同应用程序依赖的 war 包使用了相同名称的类,但是这些类版本不同。如果使用双亲委派机制,只能加载一个,不能把不同版本的都加载进去。
再详细解释:
我们都用过 Tomcat 服务器,一个 Tomcat 服务器上可能同时部署多个应用程序,这多个应用程序可能使用的第三方类库相同,但是版本不同,也就是说存在类名相同,但是内容不同的类。这时,假如我们使用了双亲委派机制,那就只能加载一个进去。很显然,这是不合理的,因此 Tomcat 使用自定义类加载器,而不使用双亲委派机制。
Tomcat 通过自定义类加载器的方式解决上述问题,其示意图如下: