Java使用ClassLoader重新加载代码

时间:2010-06-20 20:48:35

标签: java class classloader reloading

我最近发了一篇帖子Update Java code during runtime,经过几个小时摆弄不同的示例代码和阅读教程后,我遇到了以下问题:

通过使用ClassLoader,我可以在运行时使用http://www.exampledepot.com/egs/java.lang/reloadclass.html处的代码将本地变量从类MyVar1更改为类MyVar2,但我无法替换它类MyVar2,其他版本为MyVar2

MyVar1MyVar2都实现了接口VarInterface。主类使用类型VarInterface保存变量的实例。

我已经阅读了其他几个声称是正确的实现,但我无法让它工作。谁能看到我在这里做错了什么?

主类循环:

    while(true){  
        i++;  
        Thread.sleep(1000);  
        ui.ping();  
        if(i > 3)  
            replaceVar();  
    }

replaceVar:

ClassLoader parentClassLoader = MyClassLoader.class.getClassLoader();
    MyClassLoader classLoader = new MyClassLoader(parentClassLoader);
    Class newClass = classLoader.loadClass("MyVar2");
    ui = (VarInterface)newClass.newInstance();

MyClassLoader.loadClass:

public Class<?> loadClass(String what){
    // Get the directory (URL) of the reloadable class
    URL[] urls = null;
    try {
        // Convert the file object to a URL
        File dir = new File(System.getProperty("user.dir")
                +File.separator+"dir"+File.separator);
        URL url = dir.toURL();
        urls = new URL[]{url};
    } catch (MalformedURLException e) {
    }

    // Create a new class loader with the directory
    ClassLoader cl = new URLClassLoader(urls);

    // Load in the class
    Class cls = null;
    try {
        cls = cl.loadClass("MyVar2");
    } catch (ClassNotFoundException e) {
        e.printStackTrace();
    }

    return cls;
}

对于前3次迭代MyVar1.ping()被调用,之后MyVar2.ping()被无限调用,即使我替换了MyVar2.classMyVar2.java文件。

2 个答案:

答案 0 :(得分:1)

我猜你正在使用两个类加载器。一个是系统类加载器,其中包含大量代码。您的自定义类加载器使用系统类加载器作为其父级。您正在尝试使用自定义类加载器来替换系统类加载器中的类。实际上,自定义类加载器将优先委托给父类加载它自己的类。

您需要做的是确保系统类加载器不加载该类的实现。创建一个类加载器,用于加载类的实现并委托给系统类加载器。要更改实现,请创建类加载器的另一个实例。

(使用java.net.URLClassLoader.newInstance可能更容易创建自己的类加载器实现。)

答案 1 :(得分:1)

我设法解决了这个问题。我忘了将文件保留在类路径之外,这使得ClassLoader只有在实现相同接口的不同类时才会读取新实例,而不是在我用不同版本替换类时。

基本上:通过从项目中删除文件并在外部编辑它们来解决问题。