類加載器
每種類加載器只能加載指定路徑下的jar包。Java一直保持着三層類加載器的架構。
Bootstrap ClassLoader
啓動類加載器。負責加載<JAVA_HOME>\lib目錄中或-Xbootclasspath參數所指定的路徑中存放的類庫。比如rt.jar。
Extention ClassLoader
擴展類加載器。負責加載<JAVA_HOME>\lib\ext目錄中或者java.ext.dirs系統變量所指定的路徑中所有的類庫。
Appclass Loader
應用程序類加載器或系統類加載器。負責加載classpath的所有類庫,比如該應用依賴的二方包、三方包等。
雙親委派模型
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();
// 在父類加載器無法加載時,調用本身的findClass方法來進行類加載
// 其實現在URLClassLoader中
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;
}
}
從loadclass方法的分析,可以得出雙親委派模型的基本流程:一個類被加載時,首先會委派給父類加載去加載,直到啓動類加載器。只有當父加載器無法加載時,子加載器纔會加載。