在Bootstrap中我們可以看到有如下三個classloader
1、為什么要設計多個類加載器?
如果所有的類都使用一個類加載器來加載,會出現什么問題呢?
假如我們自己編寫一個類java.util.Object,它的實現可能有一定的危險性或者隱藏的bug。而我們知道Java自帶的核心類里面也有java.util.Object,如果JVM啟動的時候先行加載的是我們自己編寫的java.util.Object,那么就有可能出現安全問題!所以,Sun(后被Oracle收購)采用了另外一種方式來保證最基本的、也是最核心的功能不會被破壞。你猜的沒錯,那就是雙親委派模式!
2、什么是雙親委派模型?
雙親委派模型解決了類錯亂加載的問題,也設計得非常精妙。雙親委派模式對類加載器定義了層級,每個類加載器都有一個父類加載器。在一個類需要加載的時候,首先委派給父類加載器來加載,而父類加載器又委派給祖父類加載器來加載,以此類推。如果父類及上面的類加載器都加載不了,那么由當前類加載器來加載,并將被加載的類緩存起來。
所以上述類是這么加載的
Java自帶的核心類 -- 由啟動類加載器加載Java支持的可擴展類 -- 由擴展類加載器加載我們自己編寫的類 -- 默認由應用程序類加載器或其子類加載
為什么Tomcat的類加載器也不是雙親委派模型?
Java默認的類加載機制是通過雙親委派模型來實現的,而Tomcat實現的方式又和雙親委派模型有所區別。
原因在于一個Tomcat容器允許同時運行多個Web程序,每個Web程序依賴的類又必須是相互隔離的。因此,如果Tomcat使用雙親委派模式來加載類的話,將導致Web程序依賴的類變為共享的。
舉個例子,假如我們有兩個Web程序,一個依賴A庫的1.0版本,另一個依賴A庫的2.0版本,他們都使用了類xxx.xx.Clazz,其實現的邏輯因類庫版本的不同而結構完全不同。那么這兩個Web程序的其中一個必然因為加載的Clazz不是所使用的Clazz而出現問題!而這對于開發來說是非常致命的!