深入理解JVM - 類加載器

類加載器

每種類加載器只能加載指定路徑下的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方法的分析,可以得出雙親委派模型的基本流程:一個類被加載時,首先會委派給父類加載去加載,直到啓動類加載器。只有當父加載器無法加載時,子加載器纔會加載。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章