自定義類加載器以及打破雙親委派模型解析
1 自定義類加載器
自定義類加載器的代碼很簡單,只需要繼承ClassLoader類,覆寫findClass方法即可
其默認(rèn)實(shí)現(xiàn)是會拋出一個(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);
}
}這里是會讀取指定的類路徑classPath下的class文件。
相應(yīng)的測試代碼如下所示:
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ù)雙親委派模型,該類會由AppClassLoader來加載,不會由自定義的的MyClassLoader來進(jìn)行加載,最后運(yùn)行測試代碼
結(jié)果如下:

2 打破雙親委派模型
在像一些Tomcat的源碼中,WebappClassLoader會打破雙親委派機(jī)制。這里我們也來簡單模擬一下。
實(shí)現(xià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不用再向上去找類加載器,只會在本類中處理,這樣就打破了雙親委派模型。
然后因?yàn)檫\(yùn)行時(shí)需要加載Object類,所以將Object.class文件復(fù)制到D:/test目錄下
如下所示:

隨后再次運(yùn)行測試類
結(jié)果如下:

可以看到,java.lang包的代碼禁止被自定義的類加載器加載,防止核心API被篡改。
這是Java內(nèi)部的安全檢查機(jī)制。這里我們這種寫法是將所有的類都交由MyClassLoader來處理,所以無法加載Java核心的類庫,但是Tomcat中的類加載機(jī)制只是自定義的WebappClassLoader和CommonClassLoader打破了雙親委派模型,而其上面的BootstrapClassLoader、ExtensionClassLoader和AppClassLoader仍然還是會走雙親委派的,所以不會有問題。
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
SpringBoot Entity中枚舉類型詳細(xì)使用介紹
本文介紹SpringBoot如何在Entity(DAO)中使用枚舉類型。(本文使用MyBatis-Plus)。在實(shí)際開發(fā)中,經(jīng)常會遇到表示類型或者狀態(tài)的情況,比如:有三種支付方式:微信、支付寶、銀聯(lián)。本文介紹如何這種場景的方案對比,并用實(shí)例來介紹如何用枚舉這種最優(yōu)雅的來表示2022-10-10
Spring Security實(shí)現(xiàn)禁止用戶重復(fù)登陸的配置原理
這篇文章主要介紹了Spring Security實(shí)現(xiàn)禁止用戶重復(fù)登陸的配置原理,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2019-12-12
Java生產(chǎn)1-100的隨機(jī)數(shù)簡單實(shí)例(分享)
下面小編就為大家?guī)硪黄狫ava生產(chǎn)1-100的隨機(jī)數(shù)簡單實(shí)例(分享)。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2017-05-05
Java實(shí)現(xiàn)在線考試系統(tǒng)與設(shè)計(jì)(學(xué)生功能)
這篇文章主要介紹了Java實(shí)現(xiàn)在線考試系統(tǒng)與設(shè)計(jì)(學(xué)生功能),本文通過實(shí)例代碼給大家介紹的非常詳細(xì),具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-02-02
詳解SpringCloud服務(wù)認(rèn)證(JWT)
本篇文章主要介紹了SpringCloud服務(wù)認(rèn)證(JWT),小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2018-01-01
java 獲取服務(wù)器真實(shí)IP的實(shí)例
這篇文章主要介紹了java 獲取服務(wù)器真實(shí)IP的實(shí)例的相關(guān)資料,這里提供實(shí)現(xiàn)方法幫助大家學(xué)習(xí)理解這部分內(nèi)容,需要的朋友可以參考下2017-08-08
Java struts2 package元素配置及實(shí)例解析
這篇文章主要介紹了Java struts2 package元素配置及實(shí)例解析,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-11-11

