Java面试Offer直通车


文章目录


JVM - 实现自定义的ClassLoader就是这么简单_自定义

Pre

​JVM-白话聊一聊JVM类加载和双亲委派机制源码解析​


自定义类加载器

我们了解了双亲委派机制后,那自定义类加载器相对就很容易了 , 只需要继承 java.lang.ClassLoader 类 ,重写findClass方法即可

该类有两个核心方法:

  • 一个是loadClass(String, boolean),实现了双亲委派机制 .

  • 一个方法是findClass,默认实现是空方法

所以自定义类加载器主要是重写findClass方法


演示

JVM - 实现自定义的ClassLoader就是这么简单_自定义类加载器_02


Step1 : 复制一个Boss1 的类,编译后,取到Boss1.class 备用


Step2 : 删除Boss1类


Step3 : 编写自定义ClassLoader

JVM - 实现自定义的ClassLoader就是这么简单_java_03

抽象类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
* @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;
}

@Override
protected 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,其中会把自定义类加载器的父加载器设置为应用程序类加载器AppClassLoader
MyClassLoader 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

JVM - 实现自定义的ClassLoader就是这么简单_java_04


Step 5 : 运行结果

JVM - 实现自定义的ClassLoader就是这么简单_jvm_05


注意事项

Boss1 生成class后,需要把Boss1 删掉,不然双亲委派(我们并没有重写loadClass方法),它又从AppClassLoader加载了 。 需要确保你自定义加载的Boss1 在其父加载器中都不存在。