如何在java中运行正在运行的应用程序中的类?

Pra*_*era 20 java classloader

说我有一个名为的班级NameGenerator.我可以使用它来根据给定的逻辑生成名称.然后我TestNameGeneration用一个方法编写一个类,该方法要求用户写一封信并按照生成名称.现在我想更改NameGeneration类中的逻辑并应用该特定更改而不停止应用程序.

我这样做是为了更多地了解类加载器,有人可以解释一下我必须学习做的事情或网站任何参考的关键概念吗?

Evg*_*eev 30

这是一个工作测试.每隔5秒Test.main()从文件系统重新加载test.Test1.class并调用Test1.hello()

package test;

public class Test1 {
    public void hello() {
        System.out.println("Hello !");
    }
}

public class Test {

    static class TestClassLoader extends ClassLoader {
        @Override
        public Class<?> loadClass(String name) throws ClassNotFoundException {
            if (name.equals("test.Test1")) {
                try {
                    InputStream is = Test.class.getClassLoader().getResourceAsStream("test/Test1.class");
                    byte[] buf = new byte[10000];
                    int len = is.read(buf);
                    return defineClass(name, buf, 0, len);
                } catch (IOException e) {
                    throw new ClassNotFoundException("", e);
                }
            }
            return getParent().loadClass(name);
        }
    }

    public static void main(String[] args) throws Exception {
        for (;;) {
            Class cls = new TestClassLoader().loadClass("test.Test1");
            Object obj = cls.newInstance();
            cls.getMethod("hello").invoke(obj);
            Thread.sleep(5000);
        }
    }
}
Run Code Online (Sandbox Code Playgroud)

运行.然后更改并重新编译Test1

System.out.println("Hello !!!");
Run Code Online (Sandbox Code Playgroud)

测试正在运行.您将看到Test1.hello输出已更改

...
Hello !
Hello !
Hello !!!
Hello !!!
Run Code Online (Sandbox Code Playgroud)

这就是Tomcat重新加载webapps的方式.它为每个webapp都有一个单独的ClassLoader,并在新的ClassLoader中加载新版本.旧的GCed就像任何Java对象以及旧类一样.

请注意,我们使用TestClassLoader加载Test1并使用反射调用其第一个方法.但是所有Test1依赖项都将使用Test1类加载器进行隐式加载,即所有Test1应用程序都将由JVM加载到TestClassLoader中.