
重写findClass方法是为了在不破坏双亲委派模型的前提下实现自定义类加载逻辑,如从网络、数据库或加密文件加载字节码,并通过defineClass将字节码转为Class对象。
在Java中,重写findClass方法通常是为了实现自定义类加载逻辑,常见于继承ClassLoader的子类。默认情况下,ClassLoader的loadClass方法会调用findClass来查找并加载类。JVM推荐的做法是:让父类加载器先尝试加载类,只有在父类加载器无法完成时,才由当前类加载器通过重写的findClass方法来加载。
findClass 是 ClassLoader 中的一个受保护方法,设计初衷就是让开发者在自定义类加载器时重写它。这样可以确保双亲委派模型不被破坏,同时又能灵活控制类的获取方式,比如从网络、数据库、加密文件等来源加载字节码。
以下是实现步骤和示例:
ClassLoader 或更常用的 URLClassLoader(如果不需要完全自定义)findClass(String name) 方法defineClass() 将字节码转换为 Class 对象public class CustomClassLoader extends ClassLoader {
private String classPath;
public CustomClassLoader(String classPath) {
this.classPath = classPath;
}
@Override
protected Class> findClass(String name) throws ClassNotFoundException {
byte[] classData = loadClassData(name);
if (classData == null) {
throw new ClassNotFoundException();
}
return defineClass(name, classData, 0, classData.length);
}
private byte[] loadClassData(String className) {
// 将类名转换为路径
String fileName = classPath + File.separatorChar +
className.replace('.', File.separatorChar) + ".class";
try (InputStream is = new FileInputStream(fileName);
ByteArrayOutputStream baos = new ByteArrayOutputStream()) {
int ch;
while ((ch = is.read()) != -1) {
baos.write(ch);
}
return baos.toByteArray();
} catch (IOException e) {
return null;
}
}
}
使用方式:
CustomClassLoader loader = new CustomClassLoader("myclasses");
Class> clazz = loader.loadClass("com.example.Hello");
Object obj = clazz.newInstance();
重写时需注意以下几点:
loadClass 方法,除非你明确要打破双亲委派机制。否则应只重写 findClass
defineClass 是 ClassLoader 提供的方法,用于将字节数组转为 Class 实例,不能直接调用,必须在 findClass 或其调用链中使用NoClassDefFoundError 或安全异常基本上就这些。重写 findClass 是实现自定义类加载的核心方式,关键在于提供正确的字节码并交给 defineClass 处理,同时保持类加载的委托机制不变。不复杂但容易忽略细节。