以下是我的自定义类加载器类。我使用以下javaargs将其设置为默认的类加载器。
-Djava.system.class.loader=MyCustomClassLoader
import java.io.*;
public class MyCustomClassLoader extends ClassLoader {
public MyCustomClassLoader() { super(); }
public MyCustomClassLoader(ClassLoader parent) { super(parent); }
protected Class loadClass(String name, boolean resolve ) throws ClassNotFoundException {
Class c = super.loadClass(name, resolve );
System.out.println( "name: " + name + ", class: " + c);
return c;
}
}
此时,启动服务器时,所有类都加载了上面的自定义类加载器类。我希望根据需要更新/重新加载属于特定包(例如com.test)的类的类定义。我该怎么办?
答案 0 :(得分:0)
你应该创建一个类加载器,它将自己加载custor类,不要超级加载它们。试试我的例子,它可能有助于理解
package test;
import java.io.IOException;
import java.net.MalformedURLException;
import java.nio.file.Files;
import java.nio.file.Paths;
import java.util.HashMap;
import java.util.Map;
class MyClassLoader extends ClassLoader {
private Map<String, Class<?>> loadedClasses = new HashMap<>();
private String dir;
public MyClassLoader(String dir) throws MalformedURLException {
this.dir = dir;
}
protected Class<?> loadClass(String name, boolean resolve) throws ClassNotFoundException {
Class<?> c = findLoadedClass(name);
if (c != null) {
return c;
}
c = loadedClasses.get(name);
if (c != null) {
return c;
}
try {
String path = dir + name.replace(".","/") + ".class";
byte[] bytes = Files.readAllBytes(Paths.get(path));
c = defineClass(name, bytes, 0, bytes.length, null);
if (resolve) {
resolveClass(c);
}
return c;
} catch (IOException ex) {
return super.loadClass(name, resolve);
}
}
}
public class ChildFirstClassLoader {
public static void main(String[] args) throws Exception {
ClassLoader cl1 = new MyClassLoader("target/classes/");
Object x1 = cl1.loadClass("test.X1").newInstance();
System.out.println(x1.getClass().getClassLoader());
cl1 = null;
ClassLoader cl2 = new MyClassLoader("target/classes/");
x1 = cl2.loadClass("test.X1").newInstance();
System.out.println(x1.getClass().getClassLoader());
}
}