自定義類加載器以及打破雙親委派模型解析
1 自定義類加載器
自定義類加載器的代碼很簡(jiǎn)單,只需要繼承ClassLoader類,覆寫findClass方法即可
其默認(rèn)實(shí)現(xiàn)是會(huì)拋出一個(gè)異常:
import java.io.FileInputStream; public class MyClassLoader extends ClassLoader { private String classPath; public MyClassLoader(String classPath) { this.classPath = classPath; } private byte[] loadByte(String name) throws Exception { name = name.replaceAll("\\.", "/"); FileInputStream fis = new FileInputStream(classPath + "/" + name + ".class"); int len = fis.available(); byte[] data = new byte[len]; fis.read(data); fis.close(); return data; } @Override protected Class<?> findClass(String name) { byte[] data = new byte[0]; try { data = loadByte(name); } catch (Exception e) { e.printStackTrace(); } return defineClass(name, data, 0, data.length); } }
這里是會(huì)讀取指定的類路徑classPath下的class文件。
相應(yīng)的測(cè)試代碼如下所示:
public class MyClassLoaderTest { public static void main(String[] args) throws Exception { MyClassLoader classLoader = new MyClassLoader("D:/test"); Class clazz = classLoader.loadClass("com.hys.test.User"); System.out.println(clazz.getClassLoader().getClass().getName()); } }
這里以User類代碼為例,將其class文件放到D:/test指定目錄下:
隨后需要注意的是,需要將當(dāng)前工作空間中的User.java文件刪除。
如果不刪除,根據(jù)雙親委派模型,該類會(huì)由AppClassLoader來加載,不會(huì)由自定義的的MyClassLoader來進(jìn)行加載,最后運(yùn)行測(cè)試代碼
結(jié)果如下:
2 打破雙親委派模型
在像一些Tomcat的源碼中,WebappClassLoader會(huì)打破雙親委派機(jī)制。這里我們也來簡(jiǎn)單模擬一下。
實(shí)現(xiàn)代碼依然很簡(jiǎn)單,只需要在上述MyClassLoader類中覆寫loadClass方法即可,如下:
@Override protected Class<?> loadClass(String name, boolean resolve) throws ClassNotFoundException { synchronized (getClassLoadingLock(name)) { Class<?> c = findLoadedClass(name); if (c == null) { c = findClass(name); } if (resolve) { resolveClass(c); } return c; } }
這里loadClass方法的代碼使用的是父類ClassLoader的源碼,然后把其中使用雙親委派的代碼刪掉,這樣MyClassLoader不用再向上去找類加載器,只會(huì)在本類中處理,這樣就打破了雙親委派模型。
然后因?yàn)檫\(yùn)行時(shí)需要加載Object類,所以將Object.class文件復(fù)制到D:/test目錄下
如下所示:
隨后再次運(yùn)行測(cè)試類
結(jié)果如下:
可以看到,java.lang包的代碼禁止被自定義的類加載器加載,防止核心API被篡改。
這是Java內(nèi)部的安全檢查機(jī)制。這里我們這種寫法是將所有的類都交由MyClassLoader來處理,所以無法加載Java核心的類庫,但是Tomcat中的類加載機(jī)制只是自定義的WebappClassLoader和CommonClassLoader打破了雙親委派模型,而其上面的BootstrapClassLoader、ExtensionClassLoader和AppClassLoader仍然還是會(huì)走雙親委派的,所以不會(huì)有問題。
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
通過代理類實(shí)現(xiàn)java連接數(shù)據(jù)庫(使用dao層操作數(shù)據(jù))實(shí)例分享
java通過代理類實(shí)現(xiàn)數(shù)據(jù)庫DAO操作代碼分享,大家參考使用吧2013-12-12spring.mvc.servlet.load-on-startup屬性方法源碼解讀
這篇文章主要介紹了spring.mvc.servlet.load-on-startup的屬性方法源碼解讀,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-12-12Java設(shè)計(jì)模式之適配器模式的實(shí)現(xiàn)
這篇文章主要介紹了Java設(shè)計(jì)模式之適配器模式的實(shí)現(xiàn),適配器模式(Adapter Pattern)是作為兩個(gè)不兼容的接口之間的橋梁,這種類型的設(shè)計(jì)模式屬于結(jié)構(gòu)型模式,它結(jié)合了兩個(gè)獨(dú)立接口的功能,需要的朋友可以參考下2023-11-11Java字符編碼簡(jiǎn)介_動(dòng)力節(jié)點(diǎn)Java學(xué)院整理
這篇文章主要介紹了Java字符編碼簡(jiǎn)介,本文主要包括以下幾個(gè)方面:編碼基本知識(shí),Java,系統(tǒng)軟件,url,工具軟件等,感興趣的朋友一起看看吧2017-08-08關(guān)于Java8的foreach中使用return/break/continue產(chǎn)生的問題
這篇文章主要介紹了關(guān)于Java8的foreach()中使用return/break/continue產(chǎn)生的問題,在使用foreach()處理集合時(shí)不能使用break和continue這兩個(gè)方法,也就是說不能按照普通的for循環(huán)遍歷集合時(shí)那樣根據(jù)條件來中止遍歷,需要的朋友可以參考下2023-10-10mybatis主從表關(guān)聯(lián)查詢,返回對(duì)象帶有集合屬性解析
這篇文章主要介紹了mybatis主從表關(guān)聯(lián)查詢,返回對(duì)象帶有集合屬性解析,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-03-03SpringBoot獲取客戶端的IP地址的實(shí)現(xiàn)示例
在Web應(yīng)用程序中,獲取客戶端的IP地址是一項(xiàng)非常常見的需求,本文主要介紹了SpringBoot獲取客戶端的IP地址的實(shí)現(xiàn)示例,具有一定的參考價(jià)值,感興趣的可以了解一下2023-09-09Java連接合并2個(gè)數(shù)組(Array)的5種方法例子
最近在寫代碼時(shí)遇到了需要合并兩個(gè)數(shù)組的需求,突然發(fā)現(xiàn)以前沒用過,于是研究了一下合并數(shù)組的方式,這篇文章主要給大家介紹了關(guān)于Java連接合并2個(gè)數(shù)組(Array)的5種方法,需要的朋友可以參考下2023-12-12Java使用正則表達(dá)式判斷獨(dú)立字符的存在(代碼示例)
通過使用正則表達(dá)式,我們可以更加靈活地判斷字符串中是否包含特定的字符,并且可以控制匹配的條件,如獨(dú)立的字符,這為我們處理字符串提供了更多的選擇和功能,這篇文章主要介紹了Java使用正則表達(dá)式判斷獨(dú)立字符的存在,需要的朋友可以參考下2023-10-10