JVM - 实现自定义的ClassLoader就是这么简单
Java面試Offer直通車
文章目錄
- Pre
- 自定義類加載器
- 演示
- Step1 : 復(fù)制一個(gè)Boss1 的類,編譯后,取到Boss1.class 備用
- Step2 : 刪除Boss1類
- Step3 : 編寫自定義ClassLoader
- Step 4: 自定義目錄存放Boss1.class
- Step 5 : 運(yùn)行結(jié)果
- 注意事項(xiàng)
Pre
JVM-白話聊一聊JVM類加載和雙親委派機(jī)制源碼解析
自定義類加載器
我們了解了雙親委派機(jī)制后,那自定義類加載器相對(duì)就很容易了 , 只需要繼承 java.lang.ClassLoader 類 ,重寫findClass方法即可
該類有兩個(gè)核心方法:
-
一個(gè)是loadClass(String, boolean),實(shí)現(xiàn)了雙親委派機(jī)制 .
-
一個(gè)方法是findClass,默認(rèn)實(shí)現(xiàn)是空方法
所以自定義類加載器主要是重寫findClass方法
演示
Step1 : 復(fù)制一個(gè)Boss1 的類,編譯后,取到Boss1.class 備用
Step2 : 刪除Boss1類
Step3 : 編寫自定義ClassLoader
抽象類CLassLoader ,我們自定義的ClassLoader只需要繼承抽象類ClassLoader,重寫loadClass方法
package com.gof.facadePattern;import java.io.FileInputStream; import java.lang.reflect.Method;/*** @author 小工匠* @version v1.0* @create 2020-06-11 23:09* @motto show me the code ,change the word* @blog https://artisan.blog.csdn.net/* @description**/public class MyClassLoaderTest {static 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;}@Overrideprotected Class<?> findClass(String name) throws ClassNotFoundException {try {byte[] data = loadByte(name);//defineClass將一個(gè)字節(jié)數(shù)組轉(zhuǎn)為Class對(duì)象,這個(gè)字節(jié)數(shù)組是class文件讀取后最終的字節(jié)數(shù)組。return defineClass(name, data, 0, data.length);} catch (Exception e) {e.printStackTrace();throw new ClassNotFoundException();}}}public static void main(String args[]) throws Exception {//初始化自定義類加載器,會(huì)先初始化父類ClassLoader,其中會(huì)把自定義類加載器的父加載器設(shè)置為應(yīng)用程序類加載器AppClassLoaderMyClassLoader classLoader = new MyClassLoader("D:/artisan");//D盤創(chuàng)建 artisan/com/gof/facadePattern 目錄,將Boss類的復(fù)制類Boss1.class丟入該目錄Class clazz = classLoader.loadClass("com.gof.facadePattern.Boss1");Object obj = clazz.newInstance();// 調(diào)用sout方法Method method = clazz.getDeclaredMethod("sout", null);method.invoke(obj, null);System.out.println(clazz.getClassLoader().getClass().getName());} }defineClass 復(fù)用ClassLoader的即可 ,主要功能是將一個(gè)字節(jié)數(shù)組轉(zhuǎn)為Class對(duì)象
自定義類加載器的父加載器是AppClassLoader , 但并不是說自定義ClassLoader的父類是AppClassLoader,這一點(diǎn)一定不要搞錯(cuò)了。
Step 4: 自定義目錄存放Boss1.class
Step 5 : 運(yùn)行結(jié)果
注意事項(xiàng)
Boss1 生成class后,需要把Boss1 刪掉,不然雙親委派(我們并沒有重寫loadClass方法),它又從AppClassLoader加載了 。 需要確保你自定義加載的Boss1 在其父加載器中都不存在。
總結(jié)
以上是生活随笔為你收集整理的JVM - 实现自定义的ClassLoader就是这么简单的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: JVM-白话聊一聊JVM类加载和双亲委派
- 下一篇: Tomcat - 模拟Tomcat的we