在tomcat中重新加载类文件

时间:2012-09-07 11:34:28

标签: java tomcat classloader urlclassloader contextclassloader

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

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

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

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

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


我正在使用以下代码来检索当前的classLoader。

ClassLoader classLoader = LoggingAspect.class.getClassLoader();

下面是load class方法的实现。

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();

我仍然得到旧的实例。

请说明为什么classloader不加载修改后的类?

1 个答案:

答案 0 :(得分:2)

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

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

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

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

IService service = createService();

其中IService是定义生成的类支持的方法的接口。这样,您可以编写使用这些方法的代码而无需实际执行。