没有希望“重载”
new运算符,但是您当然可以编写一个自定义的类加载器,该类加载器在每次要求其加载类时都简单地重新加载字节码。开箱即用的类加载器不会满足您的需求,因为它们都假定类定义在JVM的整个生命周期中都不会改变。
但是,这就是您实现它的方式。创建一个称为类的类加载器,该类加载器
Reloader将覆盖方法
loadClass和
findClass方法,以便每次调用它们时都从磁盘简单地重新加载类文件(而不是“缓存”它们以供以后使用)。然后,您只要在
newReloader().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方法覆盖文件以包含一个新的实现,那么您应该每次都看到该新实现。



