JVM - 实现自定义的ClassLoader就是这么简单
生活随笔
收集整理的这篇文章主要介绍了
JVM - 实现自定义的ClassLoader就是这么简单
小编觉得挺不错的,现在分享给大家,帮大家做个参考.
Java面试Offer直通车
文章目录
- Pre
- 自定义类加载器
- 演示
- Step1 : 复制一个Boss1 的类,编译后,取到Boss1.class 备用
- Step2 : 删除Boss1类
- Step3 : 编写自定义ClassLoader
- Step 4: 自定义目录存放Boss1.class
- Step 5 : 运行结果
- 注意事项
Pre
JVM-白话聊一聊JVM类加载和双亲委派机制源码解析
自定义类加载器
我们了解了双亲委派机制后,那自定义类加载器相对就很容易了 , 只需要继承 java.lang.ClassLoader 类 ,重写findClass方法即可
该类有两个核心方法:
-
一个是loadClass(String, boolean),实现了双亲委派机制 .
-
一个方法是findClass,默认实现是空方法
所以自定义类加载器主要是重写findClass方法
演示
Step1 : 复制一个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将一个字节数组转为Class对象,这个字节数组是class文件读取后最终的字节数组。return defineClass(name, data, 0, data.length);} catch (Exception e) {e.printStackTrace();throw new ClassNotFoundException();}}}public static void main(String args[]) throws Exception {//初始化自定义类加载器,会先初始化父类ClassLoader,其中会把自定义类加载器的父加载器设置为应用程序类加载器AppClassLoaderMyClassLoader classLoader = new MyClassLoader("D:/artisan");//D盘创建 artisan/com/gof/facadePattern 目录,将Boss类的复制类Boss1.class丢入该目录Class clazz = classLoader.loadClass("com.gof.facadePattern.Boss1");Object obj = clazz.newInstance();// 调用sout方法Method method = clazz.getDeclaredMethod("sout", null);method.invoke(obj, null);System.out.println(clazz.getClassLoader().getClass().getName());} }defineClass 复用ClassLoader的即可 ,主要功能是将一个字节数组转为Class对象
自定义类加载器的父加载器是AppClassLoader , 但并不是说自定义ClassLoader的父类是AppClassLoader,这一点一定不要搞错了。
Step 4: 自定义目录存放Boss1.class
Step 5 : 运行结果
注意事项
Boss1 生成class后,需要把Boss1 删掉,不然双亲委派(我们并没有重写loadClass方法),它又从AppClassLoader加载了 。 需要确保你自定义加载的Boss1 在其父加载器中都不存在。
总结
以上是生活随笔为你收集整理的JVM - 实现自定义的ClassLoader就是这么简单的全部内容,希望文章能够帮你解决所遇到的问题。
- 上一篇: JVM-白话聊一聊JVM类加载和双亲委派
- 下一篇: Tomcat - 模拟Tomcat的we