java 詳解類加載器的雙親委派及打破雙親委派
java 詳解類加載器的雙親委派及打破雙親委派
一般的場(chǎng)景中使用Java默認(rèn)的類加載器即可,但有時(shí)為了達(dá)到某種目的又不得不實(shí)現(xiàn)自己的類加載器,例如為了達(dá)到類庫(kù)的互相隔離,例如為了達(dá)到熱部署重加載功能。這時(shí)就需要自己定義類加載器,每個(gè)類加載器加載各自的類庫(kù)資源,以此達(dá)到資源隔離效果。在對(duì)資源的加載上可以沿用雙親委派機(jī)制,也可以打破雙親委派機(jī)制。
一、沿用雙親委派機(jī)制自定義類加載器很簡(jiǎn)單,只需繼承ClassLoader類并重寫findClass方法即可。如下例子:
①先定義一個(gè)待加載的類Test,它很簡(jiǎn)單,只是在構(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對(duì)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)無(wú)法加載時(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(); } } }
感謝閱讀,希望能幫助到大家,謝謝大家對(duì)本站的支持!
相關(guān)文章
SpringBoot Actuator監(jiān)控的項(xiàng)目實(shí)踐
本文主要結(jié)合 Spring Boot Actuator,跟大家一起分享微服務(wù)Spring Boot Actuator 的常見用法,方便我們?cè)谌粘V袑?duì)我們的微服務(wù)進(jìn)行監(jiān)控治理,感興趣的可以了解一下2024-01-01Springboot結(jié)合@validated優(yōu)化代碼驗(yàn)證
這篇文章主要介紹了Springboot與@validated注解結(jié)合從而實(shí)現(xiàn)讓你的代碼驗(yàn)證更清爽,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2022-08-08Java實(shí)現(xiàn)簡(jiǎn)易版聯(lián)網(wǎng)坦克對(duì)戰(zhàn)小游戲(附源碼)
這篇文章主要給大家介紹了關(guān)于Java實(shí)現(xiàn)簡(jiǎn)易版聯(lián)網(wǎng)坦克對(duì)戰(zhàn)小游戲的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家學(xué)習(xí)或者使用java具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-04-04spring-boot-maven-plugin:unknown的完美解決方法
這篇文章主要介紹了spring-boot-maven-plugin:unknown的完美解決方法,本文通過圖文并茂的形式給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-11-11SpringBoot集成Caffeine緩存的實(shí)現(xiàn)步驟
Caffeine cache是一個(gè)針對(duì)Java的高性能緩存庫(kù)。在本文中,我們將介紹它與Spring Boot如何一起使用。2021-05-05Java HtmlEmail 郵件發(fā)送的簡(jiǎn)單實(shí)現(xiàn)代碼
下面小編就為大家?guī)?lái)一篇Java HtmlEmail 郵件發(fā)送的簡(jiǎn)單實(shí)現(xiàn)代碼。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過來(lái)看看吧2016-06-06