java - 所有類都是由Object類的記載器加載的嗎?
問題描述
看到網(wǎng)上這么說:
雙親委派機制描述 某個特定的類加載器在接到加載類的請求時,首先將加載任務(wù)委托給父類加載器,依次遞歸,如果父類加載器可以完成類加載任務(wù),就成功返回;只有父類加載器無法完成此加載任務(wù)時,才自己去加載。
首先將加載任務(wù)委托給父類加載器,依次遞歸 這句話的意思是所有類都是由Object類的記載器加載的嗎?
問題解答
回答1:Java 虛擬機的第一個類加載器是 Bootstrap,這個加載器是嵌套在 Java 虛擬機內(nèi)核里面,它是用 C++ 寫的二進制代碼(不是字節(jié)碼)。
使用委托機制,會遞歸的向父類查找,也就是首選用 Bootstrap 嘗試加載,如果找不到再向下。防止內(nèi)存中出現(xiàn)兩份類的字節(jié)碼。
你誤解了類和類加載器。
當類記載的時候,首先用當前線程的類加載器去加載線程中的第一個類,比如這個類是 ClassA,類加載器是 ClassLoaderA。
如果 ClassA 引用了 ClassB,則系統(tǒng)會使用 ClassLoaderA 去加載 ClassB。
現(xiàn)在有了 2 個類(簡化版,其實不止是 2 個類)。
我們考慮一種情況,ClassX 和 ClassY 已經(jīng)在內(nèi)存中加載,他們都引用了 ClassZ,那么 ClassZ 由誰加載呢?
很顯然按照上面描述的加載步驟會出現(xiàn) 2 份 ClassZ:ClassX 加載一次,ClassY 又加載一次。因為 ClassY 不知道 ClassX 已經(jīng)加載過了。
如何解決這個問題呢,就是向父類遞歸查找。
具體步驟就是,先從 BootstrapClassLoader 查找,如果 BootstrapClassLoader 加載了這個類,就返回,如果 BootstrapClassLoader 沒有加載過這個類,則繼續(xù)查找,直到找到這個類。如果一直找到了本線程的類加載器都沒有找到,說明這個類還沒有加載,則使用當前線程的加載器加載。可以使用 getContextClassLoader() 獲得當前線程的類加載器。
回答2:Java中有兩類類加載器:系統(tǒng)類加載器和用戶自定義類加載器。
系統(tǒng)類加載器都會有加載路徑的限定,比如Bootstrap Class Loader在JDK1.6下,通過System.getProperty('sun.boot.class.path')可以得到類加載路徑
JAVA_HOMEjre6libresources.jar; JAVA_HOMEJavajre6librt.jar; JAVA_HOMEjre6libsunrsasign.jar; JAVA_HOMEjre6libjsse.jar;JAVA_HOMEjre6libjce.jar; JAVA_HOMEjre6libcharsets.jar; JAVA_HOMEjre6libmodulesjdk.boot.jar; JAVA_HOMEjre6classes
這些路徑下的class是由Bootstrap負責(zé),其它路徑下的class的遞歸到Bootstrap下也是找不到class文件,就會由下一級類加載器去相應(yīng)的路徑去加載。
回答3:樓上說得對,建議看看jvm相關(guān)書籍,詳細了解一下.
相關(guān)文章:
1. angular.js - angular內(nèi)容過長展開收起效果2. docker不顯示端口映射呢?3. dockerfile - [docker build image失敗- npm install]4. docker - 如何修改運行中容器的配置5. 在windows下安裝docker Toolbox 啟動Docker Quickstart Terminal 失敗!6. javascript - nodejs調(diào)用qiniu的第三方資源抓取,返回401 bad token,為什么7. docker - 各位電腦上有多少個容器啊?容器一多,自己都搞混了,咋辦呢?8. docker網(wǎng)絡(luò)端口映射,沒有方便點的操作方法么?9. 為什么我ping不通我的docker容器呢???10. dockerfile - 我用docker build的時候出現(xiàn)下邊問題 麻煩幫我看一下
