1

我在运行时创建一个类文件。

我想用类加载器中的更新的类文件替换现有的类文件。

它类似于避免服务器重新启动和重新部署的热交换(例如 JRebel)。

我找到了用于上下文重新加载的 tomcat 的 context.xml 方法。但在生产环境中它不是很有用。

我们可以在运行时向 ClassLoader 注册类吗?请建议是否有任何替代方法可以在运行时重新加载类。


我正在使用以下代码来检索当前的类加载器。

ClassLoader classLoader = LoggingAspect.class.getClassLoader();

下面是加载类方法的实现。

public class AspectClassLoader extends ClassLoader{

@Override
public synchronized Class<?> loadClass(String name) throws ClassNotFoundException
{
    String customLoadClass = "com.log.LoggingAspect";
    try
    {
        if(!customLoadClass.equals(name))
        {
            return super.loadClass(name);
        }
        else
        {
            URL classLoadUrl = new URL(this.reloadClassUrl);
            URLConnection connection = classLoadUrl.openConnection();
            InputStream input = connection.getInputStream();
            ByteArrayOutputStream buffer = new ByteArrayOutputStream();
            int data = input.read();

            while(data != -1){
                buffer.write(data);
                data = input.read();
            }

            input.close();

            byte[] classData = buffer.toByteArray();
            return defineClass(name, classData, 0, classData.length);
        }
    }
    catch(Exception e)
    {
        e.printStackTrace();
    }
    return null; 
}

要重新加载课程,我正在使用以下代码。

AspectClassLoader urlClsLoader = new AspectClassLoader(classLoader);
Class aspect = urlClsLoader.reloadClass("com.log.LoggingAspect", true, new String("file:"+className));

我的类加载器的重载方法是

public synchronized Class<?> reloadClass(String name, boolean resolve, String reloadClassUrl) throws ClassNotFoundException
{
    this.reloadClassUrl = reloadClassUrl;
    return this.loadClass(name, resolve);
}

重新加载后,如果我创建 LoggingAspect 的新实例,它仍然会给我旧实例。请建议。

Class aspect = urlClsLoader.reloadClass("com.log.LoggingAspect", true, new String("file:"+className));
        com.log.aspect.Aspect aspectObj = (com.log.aspect.Aspect)aspect.newInstance();

我仍然得到旧实例。

请建议为什么类加载器不加载修改后的类?

4

1 回答 1

2

这是可能的,但很复杂。您需要做的是以URLClassLoader当前ClassLoader为父级创建一个新的。将 URL 添加到包含您的类的文件夹(没有包文件夹)到新的URLClassLoader

接下来,您必须loadClass()在调用之前修补以返回您的新类parent.loadClass()(否则,将使用现有的类而忽略更新的类)。

当您想使用新类时,它会变得复杂。为此,您必须使用新的类加载器创建它,并确保每个人都使用该类型的此实例。

如果您的 WAR 中不存在该类,它可能更安全(并且更容易调试)。不使用类型,而是使用 加载类URLClassLoader并创建它的实例。所以使用这个类的代码是这样工作的:

IService service = createService();

whereIService是一个接口,它定义了生成的类支持的方法。这样,您可以编写使用这些方法的代码,而实际上不需要实现。

于 2012-09-07T12:14:32.020 回答