8

我正在使用带有eclipse的tomcat。

根据 tomcat 的文档:

从 web 应用程序的角度来看,类或资源加载按以下顺序查找以下存储库:

  • JVM 系统类加载器类的引导类(如上所述)
  • /WEB-INF/Web 应用程序的类
  • /WEB-INF/lib/*.jar 您的 Web 应用程序
  • 通用类加载器类(如上所述)

因此,当加载一个类时,tomcat 会在WEB-INF/lib之前查找WEB-INF/classes。并且我们可以覆盖WEB-INF/lib 中jar 文件中的一些类,tomcat 会拾取被覆盖的类。

但是现在,如果我通过选中“Serve modules with out publishing”更改了 tomcat 服务器选项,那么将不再加载被覆盖的类。

有什么解决方案可以让它再次工作,但我仍然希望 tomcat 在不发布的情况下为模块提供服务。


编辑:
我发现一些可能有用的东西,在这个文件夹f:\eclipse_projects\.metadata.plugins\org.eclipse.wst.server.core\tmp0\conf 有一个 server.xml 它包含这样的内容:

<Resources className="org.eclipse.jst.server.tomcat.loader.WtpDirContext" 
    extraResourcePaths="" 
    virtualClasspath="F:\eclipse_projects\ALS7C3\bin"/>
<Loader className="org.eclipse.jst.server.tomcat.loader.WtpWebappLoader" 
    useSystemClassLoaderAsParent="false" 
    virtualClasspath="F:\eclipse_projects\ALS7C3\bin"/>

似乎在选中“服务模块而不发布”选项运行 tomcat 时,eclipse 将使用它自己的加载器。这个加载器包含在一个 jar 文件中,该文件将被复制到f:\eclipse_projects\.metadata.plugins\org.eclipse。 wst.server.core\tmp0\lib** 当你在 Eclipse 中启动 tomcat 时。这是 **org.eclipse.jst.server.tomcat.loader.WtpDirContext 的部分源代码

public Object lookup(String name) throws NamingException {
        if (name.startsWith("/WEB-INF/") && name.endsWith(".tld")) {
            String tldName = name.substring(name.lastIndexOf("/") + 1);
            if (virtualMappings != null && virtualMappings.containsKey(tldName)) {
                return new FileResource(virtualMappings.get(tldName));
            }
        } else if (tagfileMappings != null && name.startsWith("/META-INF/tags")
                && (name.endsWith(".tag") || name.endsWith(".tagx"))) {
            // already loaded tag file
            return new FileResource(tagfileMappings.get(name));
        } 

        Object retSuper;
        NamingException superEx;
        try {
            retSuper = super.lookup(name);
            return retSuper;
        } catch (NamingException ex) {
            retSuper = null;
            superEx = ex;
        }
        if (mappedResourcePaths != null) {
            // Perform lookup using the extra resource paths
            for (Map.Entry<String, List<String>> mapping : mappedResourcePaths.entrySet()) {
                String path = mapping.getKey();
                List<String> dirList = mapping.getValue();
                if (name.equals(path)) {
                    for (String resourcesDir : dirList) {
                        File f = new File(resourcesDir);
                        if (f.exists() && f.canRead()) {
                            if (f.isFile()) {
                                return new FileResource(f);
                            }
                            else {
                                // TODO Handle directory
                            }
                        }
                    }
                }
                path += "/";
                if (name.startsWith(path)) {
                    String res = name.substring(path.length());
                    for (String resourcesDir : dirList) {
                        File f = new File (resourcesDir + "/" + res);
                        if (f.exists() && f.canRead()) {
                            if (f.isFile()) {
                                return new FileResource(f);
                            }
                            else {
                                // TODO Handle directory
                            }
                        }
                    }
                }
            }
        }
        throw superEx;
    }

似乎如果它首先处理 jsp 标签库然后调用super.lookup,如果在super.lookup中找不到,它将尝试在 virtualClasspath 中加载资源,在我的示例中为“F:\eclipse_projects\ALS7C3\bin”,这是 eclipse 在不发布服务模块时输出类文件的地方。

所以,我想,如果我可以覆盖org.eclipse.jst.server.tomcat.loader.WtpDirContext的查找方法,我可以得到我想要的,但是这个 jar 文件包含在org.eclipse.jst.server.tomcat 中。 core.jar,它们都已签名。

我不知道如何覆盖这样的 jar 文件。

任何人都可以帮忙吗?

4

2 回答 2

0

这是我在 eclipse/tomcat 中使用热部署的方法:

  1. 停止tomcat
  2. 打开tomcat配置菜单
  3. 在“服务器选项”选项卡中选中“服务模块而不发布”
  4. 在“发布”选项卡中选中“从不自动发布”
  5. 保存配置
  6. 以调试模式启动tomcat(这个很重要)

如果您更改了配置文件,也许可以尝试使用新的 tomcat。

于 2015-04-20T20:21:11.333 回答
0

覆盖这些类的解决方案是Servers\???-config\catalina.properties在工作区项目中进行修改。

common.loader=${catalina.base}/lib,${catalina.base}/lib/*.jar,${catalina.home}/lib,${catalina.home}/lib/*.jar

该属性common.loader是 Tomcat 常用 ClassLoader 的类路径设置。
Webtools 插件将org.eclipse.jst.server.tomcat.core.jar复制到${catalina.base}/lib. 您可以覆盖这些类并替换${catalina.base}/lib,${catalina.base}/lib/*.jar为新路径。

于 2016-05-21T11:43:12.083 回答