java 詳解類加載器的雙親委派及打破雙親委派
java 詳解類加載器的雙親委派及打破雙親委派
一般的場景中使用Java默認(rèn)的類加載器即可,但有時(shí)為了達(dá)到某種目的又不得不實(shí)現(xiàn)自己的類加載器,例如為了達(dá)到類庫的互相隔離,例如為了達(dá)到熱部署重加載功能。這時(shí)就需要自己定義類加載器,每個(gè)類加載器加載各自的類庫資源,以此達(dá)到資源隔離效果。在對資源的加載上可以沿用雙親委派機(jī)制,也可以打破雙親委派機(jī)制。
一、沿用雙親委派機(jī)制自定義類加載器很簡單,只需繼承ClassLoader類并重寫findClass方法即可。如下例子:
①先定義一個(gè)待加載的類Test,它很簡單,只是在構(gòu)建函數(shù)中輸出由哪個(gè)類加載器加載。
public class Test {
public Test(){
System.out.println(this.getClass().getClassLoader().toString());
}
}
②定義一個(gè)TestClassLoader類繼承ClassLoader,重寫findClass方法,此方法要做的事情是讀取Test.class字節(jié)流并傳入父類的defineClass方法即可。然后就可以通過自定義累加載器TestClassLoader對Test.class進(jìn)行加載,完成加載后會(huì)輸出“TestLoader”。
public class TestClassLoader extends ClassLoader {
private String name;
public TestClassLoader(ClassLoader parent, String name) {
super(parent);
this.name = name;
}
@Override
public String toString() {
return this.name;
}
@Override
public Class<?> findClass(String name) {
InputStream is = null;
byte[] data = null;
ByteArrayOutputStream baos = new ByteArrayOutputStream();
try {
is = new FileInputStream(new File("d:/Test.class"));
int c = 0;
while (-1 != (c = is.read())) {
baos.write(c);
}
data = baos.toByteArray();
} catch (Exception e) {
e.printStackTrace();
} finally {
try {
is.close();
baos.close();
} catch (IOException e) {
e.printStackTrace();
}
}
return this.defineClass(name, data, 0, data.length);
}
public static void main(String[] args) {
TestClassLoader loader = new TestClassLoader(
TestClassLoader.class.getClassLoader(), "TestLoader");
Class clazz;
try {
clazz = loader.loadClass("test.classloader.Test");
Object object = clazz.newInstance();
} catch (Exception e) {
e.printStackTrace();
}
}
}
二、打破雙親委派機(jī)制則不僅要繼承ClassLoader類,還要重寫loadClass和findClass方法,如下例子:
①定義Test類。
public class Test {
public Test(){
System.out.println(this.getClass().getClassLoader().toString());
}
}
②重新定義一個(gè)繼承ClassLoader的TestClassLoaderN類,這個(gè)類與前面的TestClassLoader類很相似,但它除了重寫findClass方法外還重寫了loadClass方法,默認(rèn)的loadClass方法是實(shí)現(xiàn)了雙親委派機(jī)制的邏輯,即會(huì)先讓父類加載器加載,當(dāng)無法加載時(shí)才由自己加載。這里為了破壞雙親委派機(jī)制必須重寫loadClass方法,即這里先嘗試交由System類加載器加載,加載失敗才會(huì)由自己加載。它并沒有優(yōu)先交給父類加載器,這就打破了雙親委派機(jī)制。
public class TestClassLoaderN extends ClassLoader {
private String name;
public TestClassLoaderN(ClassLoader parent, String name) {
super(parent);
this.name = name;
}
@Override
public String toString() {
return this.name;
}
@Override
public Class<?> loadClass(String name) throws ClassNotFoundException {
Class<?> clazz = null;
ClassLoader system = getSystemClassLoader();
try {
clazz = system.loadClass(name);
} catch (Exception e) {
// ignore
}
if (clazz != null)
return clazz;
clazz = findClass(name);
return clazz;
}
@Override
public Class<?> findClass(String name) {
InputStream is = null;
byte[] data = null;
ByteArrayOutputStream baos = new ByteArrayOutputStream();
try {
is = new FileInputStream(new File("d:/Test.class"));
int c = 0;
while (-1 != (c = is.read())) {
baos.write(c);
}
data = baos.toByteArray();
} catch (Exception e) {
e.printStackTrace();
} finally {
try {
is.close();
baos.close();
} catch (IOException e) {
e.printStackTrace();
}
}
return this.defineClass(name, data, 0, data.length);
}
public static void main(String[] args) {
TestClassLoaderN loader = new TestClassLoaderN(
TestClassLoaderN.class.getClassLoader(), "TestLoaderN");
Class clazz;
try {
clazz = loader.loadClass("test.classloader.Test");
Object object = clazz.newInstance();
} catch (Exception e) {
e.printStackTrace();
}
}
}
感謝閱讀,希望能幫助到大家,謝謝大家對本站的支持!
相關(guān)文章
SpringBoot Actuator監(jiān)控的項(xiàng)目實(shí)踐
本文主要結(jié)合 Spring Boot Actuator,跟大家一起分享微服務(wù)Spring Boot Actuator 的常見用法,方便我們在日常中對我們的微服務(wù)進(jìn)行監(jiān)控治理,感興趣的可以了解一下2024-01-01
Springboot結(jié)合@validated優(yōu)化代碼驗(yàn)證
這篇文章主要介紹了Springboot與@validated注解結(jié)合從而實(shí)現(xiàn)讓你的代碼驗(yàn)證更清爽,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2022-08-08
Java實(shí)現(xiàn)簡易版聯(lián)網(wǎng)坦克對戰(zhàn)小游戲(附源碼)
這篇文章主要給大家介紹了關(guān)于Java實(shí)現(xiàn)簡易版聯(lián)網(wǎng)坦克對戰(zhàn)小游戲的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對大家學(xué)習(xí)或者使用java具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來一起學(xué)習(xí)學(xué)習(xí)吧2019-04-04
spring-boot-maven-plugin:unknown的完美解決方法
這篇文章主要介紹了spring-boot-maven-plugin:unknown的完美解決方法,本文通過圖文并茂的形式給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-11-11
SpringBoot集成Caffeine緩存的實(shí)現(xiàn)步驟
Caffeine cache是一個(gè)針對Java的高性能緩存庫。在本文中,我們將介紹它與Spring Boot如何一起使用。2021-05-05
Java HtmlEmail 郵件發(fā)送的簡單實(shí)現(xiàn)代碼
下面小編就為大家?guī)硪黄狫ava HtmlEmail 郵件發(fā)送的簡單實(shí)現(xiàn)代碼。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2016-06-06

