您好,登錄后才能下訂單哦!
今天小編給大家分享一下Java的雙親委派模式怎么實現的相關知識點,內容詳細,邏輯清晰,相信大部分人都還太了解這方面的知識,所以分享這篇文章給大家參考一下,希望大家閱讀完這篇文章后有所收獲,下面我們一起來了解一下吧。
說起雙親委派模型,不得不說一下類加載器。
當我們編譯Java類時,JVM會創建與平臺和機器無關的字節碼。字節碼存儲在.class
文件中。當我們嘗試使用一個類時,類加載器就會把它加載到內存中,然后把字節碼文件轉成Class對象。通俗的說類加載器就是將.class
文件轉成Class對象的。
啟動類加載器(Bootstrap Class Loader):負責加載存放在 <JAVA_HOME>\lib
目錄下的類,或者被-Xbootclasspath
參數所指定的路徑中存放的類。比如:rt.jar
、java.lang.*
包下的類。
擴展類加載器(Extension Class Loader):負責加載<JAVA_HOME>\lib\ext
目錄中,或者被java.ext.dirs
系統變量所指定的路徑中所有的類庫。
應用程序類加載器(Application Class Loader):負責加載用戶類路徑上所有的類庫。
當類加載器收到加載類的請求時,它首先會把請求委派給父加載器去完成,每一層都如此,直到把請求委派給最頂層的啟動類加載器,只有當父加載器無法加載委派過來的類時,子加載器才會加載。
JVM在加載?個類時,會調?AppClassLoader的loadClass
?法來加載這個類,不過在這個?法中,會先使?ExtClassLoader的loadClass
?法來加載類,同樣ExtClassLoader的loadClass
?法中會先使?BootstrapClassLoader來加載類,如果BootstrapClassLoader加載到了就直接成功,如果 BootstrapClassLoader沒有加載到,那么ExtClassLoader就會??嘗試加載該類,如果沒有加載到,那么則會由AppClassLoader來加載這個類。
所以,雙親委派指得是,JVM在加載類時,會委派給ExtClassLoader和BootstrapClassLoader進?加載,如果沒加載到才由??進?加載。
這里說的雙親并不是說類加載器之間是以繼承方式實現的,而是以組合的方式實現的,通過源碼可以證實這點:
java.lang.ClassLoader#loadClass
private final ClassLoader parent; 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(); 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()
方法,若父加載器為空則默認使用啟動類加載器作為父加載器。假如父類加載器加載失敗,拋出ClassNotFoundException
異常的話,才調用自己的findClass()
方法嘗試進行加載。
保證唯一性,避免重復加載:類的加載隨著它的類加載器一起具備了層級關系,通過這種層級關系避免了重復加載,父類加載器加載了該類,子加載器就無需加載了。
避免核心類被篡改:核心類由啟動類加載器加載,即使用戶自定義同名核心類也不會被加載。
以上就是“Java的雙親委派模式怎么實現”這篇文章的所有內容,感謝各位的閱讀!相信大家閱讀完這篇文章都有很大的收獲,小編每天都會為大家更新不同的知識,如果還想學習更多的知識,請關注億速云行業資訊頻道。
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。