問(wèn)題場(chǎng)景
??現(xiàn)在很多工程為了功能擴(kuò)展猾警,都給出了插件化的方式孔祸。只需要用戶配置好配置文件,提供好需要的jar包发皿,就能完成相應(yīng)功能〈藁郏現(xiàn)在項(xiàng)目一般都離不開(kāi)數(shù)據(jù)庫(kù),自己本身的項(xiàng)目就會(huì)帶這驅(qū)動(dòng)包穴墅,但是也會(huì)有這樣的一種需求惶室,就是數(shù)據(jù)額外存儲(chǔ)的定制化温自,當(dāng)產(chǎn)生的數(shù)據(jù)在自己項(xiàng)目的流程中不滿足現(xiàn)在使用。例如做報(bào)表皇钞,項(xiàng)目本身產(chǎn)生數(shù)據(jù)悼泌,但是需要把里面的一部分?jǐn)?shù)據(jù)拿出來(lái)和其他文本數(shù)據(jù)結(jié)合,產(chǎn)生新的數(shù)據(jù)夹界」堇铮或者現(xiàn)有的數(shù)據(jù)進(jìn)行時(shí)間的整合,直接變成周數(shù)據(jù)可柿,月數(shù)據(jù)鸠踪,年數(shù)據(jù)。這些都是需要根據(jù)不同用戶自己設(shè)定的趾痘。這些數(shù)據(jù)往往是需要另外的數(shù)據(jù)庫(kù)的慢哈,這就帶來(lái)了一個(gè)問(wèn)題,項(xiàng)目本身有一個(gè)jar包永票,例如Mysql4.5的卵贱,客戶想組織信息存入mysql5.5,jar包的類是相同的侣集,這樣就帶來(lái)了驅(qū)動(dòng)加載的問(wèn)題键俱,因?yàn)轭惖南嗤模因?qū)動(dòng)不是完全不兼容世分,而是在使用上會(huì)出問(wèn)題编振。典型的就是ojdbc14,用這個(gè)版本的驅(qū)動(dòng)建立數(shù)據(jù)庫(kù)使用語(yǔ)句池緩存會(huì)有問(wèn)題臭埋,ojdbc5踪央,6就沒(méi)事。(畢竟語(yǔ)句池緩存能帶來(lái)性能上優(yōu)化瓢阴,不能說(shuō)為了兼容驅(qū)動(dòng)放棄性能畅蹂,而且其他潛在的問(wèn)題還沒(méi)有暴露)。
問(wèn)題分析
??雙親委派模式的是好處是Java類隨著它的類加載器一起具備了一種帶有優(yōu)先級(jí)的層次關(guān)系荣恐,通過(guò)這種層級(jí)關(guān)可以避免類的重復(fù)加載液斜,當(dāng)父親已經(jīng)加載了該類時(shí),就沒(méi)有必要子ClassLoader再加載一次叠穆。其次是考慮到安全因素少漆,java核心api中定義類型不會(huì)被隨意替換,假設(shè)通過(guò)網(wǎng)絡(luò)傳遞一個(gè)名為java.lang.Integer的類硼被,通過(guò)雙親委托模式傳遞到啟動(dòng)類加載器示损,而啟動(dòng)類加載器在核心JavaAPI發(fā)現(xiàn)這個(gè)名字的類,發(fā)現(xiàn)該類已被加載嚷硫,并不會(huì)重新加載網(wǎng)絡(luò)傳遞的過(guò)來(lái)的java.lang.Integer屎媳,而直接返回已加載過(guò)的Integer.class夺溢,這樣便可以防止核心API庫(kù)被隨意篡改。雙親委派的這些特性導(dǎo)致一個(gè)程序想要加載兩個(gè)不同版本包時(shí)烛谊,由于已近記載了一個(gè)包,另一個(gè)不回加載嘉汰。
實(shí)現(xiàn)
??首先需要一個(gè)違背雙親委托的classloader丹禀,
????????不論你是class.forname,還是loadclass方法的調(diào)用,最后都是要調(diào)用loadClass的鞋怀,所以這里一定要重寫(xiě)這個(gè)方法双泪,這里就要加入破壞雙親委托機(jī)制的邏輯。先從緩存找是否已經(jīng)加載了class密似,對(duì)已經(jīng)加載的就直接返回焙矛,防止重復(fù)加載。此處調(diào)用了findClass方法残腌,findClass是可以覆寫(xiě)的村斟,這里為了簡(jiǎn)潔的實(shí)現(xiàn),就不再覆寫(xiě)了抛猫。
????????classloader如果有重復(fù)的jar包蟆盹,findClass找到資源后會(huì)通過(guò)defineClass(String name, Resource res)來(lái)加載類,這里最后會(huì)調(diào)用getPackage來(lái)獲取包闺金。
????????這個(gè)方法逾滥,先從已經(jīng)加載的包中查找,如果沒(méi)有就先從父classloader找败匹,最后又進(jìn)行了雙親委托機(jī)制寨昙。所以這個(gè)地方也需要覆寫(xiě)。覆寫(xiě)直接返回空掀亩。在后面的判斷中舔哪,如果此處返回null,后續(xù)就會(huì)重新新建一個(gè)對(duì)象归榕,然后放入一個(gè)緩存結(jié)構(gòu)尸红,還是一個(gè)hashmap,?為空的結(jié)果只是更新一下緩存刹泄。
? ? 自定義的classloader很好的保證了我們加載的安全性外里,因?yàn)榈刂肥俏覀冎付ǖ摹?/p>
通過(guò)DriverManager加載在項(xiàng)目里的包特石,通過(guò)自定義的driver加載另一個(gè)版本的包盅蝗,兩個(gè)包之間互不影響,都可以獲取到j(luò)dbc鏈接姆蘸。