如何在实例化时强制Java重新加载类?

背景:

我有一个MainTest类,其中包含许多按钮,每个按钮都实例化我正在编码/测试的类。我希望这些类的代码/测试周期更快,并希望每分钟几次快速查看更改的效果。稳定的MainTest大约需要20秒的加载时间,如果我实例化的类中的每个更改都不需要重新加载它,这将不是问题。我想一次加载MainTest,当它实例化另一个类时,我们多次调用它为ChildTest(通过按钮事件),它应该重新加载最新版本的ChildTest。

我尝试了Class.ForName,但没有任何改变。

我也尝试使用自定义类加载器(从开源复制),无济于事。

回答:

没有希望“重载”

new运算符,但是您当然可以编写一个自定义的类加载器,该类加载器在每次要求其加载类时都简单地重新加载字节码。开箱即用的类加载器不会满足您的需求,因为它们都假定类定义在JVM的整个生命周期中都不会改变。

但是,这就是您实现它的方式。创建一个称为类的类加载器,该类加载器Reloader将覆盖方法loadClassfindClass方法,以便每次调用它们时都从磁盘简单地重新加载类文件(而不是“缓存”它们以供以后使用)。然后,您只要在new

Reloader().loadClass("foo.bar.MyClassName")怀疑类定义已更改的任何时候都必须调用(例如,作为测试框架的生命周期方法的一部分)。

本文填充了一些详细信息,但遗漏了一些要点,尤其是有关使用类加载器的新实例进行后续重新加载以及在适当时委派给默认类加载器的内容。这是一个简单的工作示例,该示例重复加载该类MyClass并假定其类文件位于相对的“

./bin”目录中:

public class Reloader extends ClassLoader {

public static void main(String[] args) throws Exception {

do {

Object foo = new Reloader().loadClass("MyFoo").newInstance();

System.out.println("LOADED: " + foo); // Overload MyFoo#toString() for effect

System.out.println("Press <ENTER> when MyFoo.class has changed");

System.in.read();

} while (true);

}

@Override

public Class<?> loadClass(String s) {

return findClass(s);

}

@Override

public Class<?> findClass(String s) {

try {

byte[] bytes = loadClassData(s);

return defineClass(s, bytes, 0, bytes.length);

} catch (IOException ioe) {

try {

return super.loadClass(s);

} catch (ClassNotFoundException ignore) { }

ioe.printStackTrace(System.out);

return null;

}

}

private byte[] loadClassData(String className) throws IOException {

File f = new File("bin/" + className.replaceAll("\\.", "/") + ".class");

int size = (int) f.length();

byte buff[] = new byte[size];

FileInputStream fis = new FileInputStream(f);

DataInputStream dis = new DataInputStream(fis);

dis.readFully(buff);

dis.close();

return buff;

}

}

在main方法中每次执行“ do /

while”块时,都会实例化一个新的Reloader,该Reloader从磁盘加载该类并将其返回给调用者。因此,如果您bin/MyClass.class使用不同的重载toString方法覆盖文件以包含一个新的实现,那么您应该每次都看到该新实现。

以上是 如何在实例化时强制Java重新加载类? 的全部内容, 来源链接: utcz.com/qa/420139.html

回到顶部