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