如何在 java 中为 Servlet 上下文添加别名?

Posted

技术标签:

【中文标题】如何在 java 中为 Servlet 上下文添加别名?【英文标题】:How do I add aliases to a Servlet Context in java? 【发布时间】:2012-09-24 18:49:56 【问题描述】:

我有一个在 Tomcat 下运行的 servlet。 我需要提供一些文件,我想我们可以从外部(到 WEB-APP)目录中将它们称为半静态的(偶尔会更改......它们由应用程序的另一部分更新)。 我设法通过将以下内容添加到 META-INF 目录中的 context.xml 来做到这一点

<Context aliases="/working_dir=c:/apache_tomcat_working_dir" ></Context>

这很好用,在我的 html 中我将文件称为

<img src="/myWebbApp/working_dir/fixpermin_zoom.png">

在我的 web.xml 里面的 WEB-INF 我让默认服务器按如下方式处理 png 文件

<!-- use default for static serving of png's, js and css, also ico -->
<servlet-mapping>
    <servlet-name>default</servlet-name>
    <url-pattern>*.png</url-pattern>
</servlet-mapping>

所以这很好用。但我想从 java 代码中设置外部目录,而不是通过编辑 context.xml 文件。

现在在 servlet 的 init() 方法中,我可以获得 ServletContext。

    ServletContext sc =  getServletContext();

如果我在调试器中检查这个变量sc,我可以看到几个层次的别名字符串,见附图。如何以编程方式获取此别名字符串? 我已经检查了 ServletContext 文档,但我找不到它很有帮助。 非常感谢任何帮助。

(来源:choicecomp.com)

【问题讨论】:

【参考方案1】:

正如您在调试器中看到的,您的上下文是 Tomcat 的上下文对象org.apache.catalina.core.StandardContext

您可以在 Tomcat 6 及以下版本中尝试以下步骤:

    StandardEngine engine = (StandardEngine) ServerFactory.getServer().findService("Catalina").getContainer();
    StandardContext context = (StandardContext) engine.findChild(engine.getDefaultHost()).findChild(getServletContext().getContextPath());
    Mapper mapper = context.getMapper();

现在您可以使用 Mapper 类的 addHostAlias(String HostName, String alias) 方法添加主机别名。

    mapper.addHostAlias(engine.getDefaultHost(), "myAlias");

这是 Tomcat 7 的代码 sn-p:

    MBeanServer mBeanServer = MBeanServerFactory.findMBeanServer(null).get(0);
    ObjectName name = new ObjectName("Catalina", "type", "Server");
    Server server = (Server) mBeanServer.getAttribute(name, "managedResource");
    StandardEngine engine = (StandardEngine) server.findService("Catalina").getContainer();
    StandardContext context = (StandardContext) engine.findChild(engine.getDefaultHost()).findChild(getServletContext().getContextPath());
    Mapper mapper = context.getMapper();
    mapper.addHostAlias(engine.getDefaultHost(), "myAlias");

如果映射器中没有主机,请尝试以下操作:

    MBeanServer mBeanServer = MBeanServerFactory.findMBeanServer(null).get(0);
    ObjectName name = new ObjectName("Catalina", "type", "Server");
    Server server = (Server) mBeanServer.getAttribute(name, "managedResource");
    StandardEngine engine = (StandardEngine) server.findService("Catalina").getContainer();
    StandardContext context = (StandardContext) engine.findChild(engine.getDefaultHost()).findChild(getServletContext().getContextPath());
    Mapper mapper = context.getMapper();
    //just a clean up step(remove the host)
    mapper.removeHost(engine.getDefaultHost());
    //add the host back with all required aliases 
    mapper.addHost(engine.getDefaultHost(), new String[]"myAlias", engine.getDefaultHost());

希望这会有所帮助!

【讨论】:

感谢您的回答,但这对我在 Tomcat 7 上不起作用。我已经检查了调试器,并且在 addHostAlias 调用之后别名部分保持不变。顺便说一句,您在哪里找到此信息?有没有关于 Tomcat 服务器内部的好教程,或者你在浏览 Tomcats 源 javadoc 还是什么? 这是工作代码。请尝试我更新答案的最后一部分。 再次感谢,但这仍然不起作用。你在哪里调用这个sn-p代码?在 servlet Init() 函数内部?你怎么知道这行得通?你能在调试器中看到上下文别名成员的变化吗?服务器是否找到您正在提供的文件?我已将虚拟名称放入我的 context.xml 文件中的别名中,然后我尝试通过您的方法添加的别名:a)不显示在调试器中,并且 b)servelet 无法从我的工作目录中提供静态文件 是的。我正在使用 Tamcat 7.0.30 并在我的测试 servlet 的 doGet 方法上调用此代码 sn-p。在我的调试器中,在执行此代码 sn-p 之前,我可以看到没有分配别名(在我的情况下,也没有主机)。执行后,我可以看到主机添加了给定的别名。【参考方案2】:

我找到了另一种方法StandardContext.setAliases。在下面找到 Tomcat 7.0.30 的完整工作代码 sn-p。

        MBeanServer mBeanServer = MBeanServerFactory.findMBeanServer(null).get(0);
        ObjectName name = new ObjectName("Catalina", "type", "Server");
        Server server = (Server) mBeanServer.getAttribute(name, "managedResource");
        StandardEngine engine = (StandardEngine) server.findService("Catalina").getContainer();
        StandardContext context = (StandardContext) engine.findChild(engine.getDefaultHost()).findChild(getServletContext().getContextPath());
        context.setAliases("myAlias");
        //infact aliases should be proper e.g. below 
        //context.setAliases("/aliasPath1=docBase1,/aliasPath2=docBase2");
        Mapper mapper = context.getMapper();
        mapper.removeHost(engine.getDefaultHost());
        mapper.addHost(engine.getDefaultHost(), new String[]"myAlias", engine.getDefaultHost());
        mapper.addHostAlias(engine.getDefaultHost(), "myAlias");
        //infact aliases should be proper e.g. below 
        //mapper.addHostAlias(engine.getDefaultHost(), "/aliasPath1=docBase1,/aliasPath2=docBase2");

请在下面找到我的调试器截图:

在代码 sn -p 执行之前:

代码sn -p执行后:

希望这更有帮助。

【讨论】:

嗨 Yogendra,感谢这个函数 context.SetAliases(... 确实改变了我在调试器中看到的别名。但是我的 servlet 仍然没有从我的别名目录中提供静态文件。我具有以前在 context.xml 文件中为我工作的相同别名字符串。我认为现在不再需要引用 mapper 的代码行?您在哪里可以找到相关信息或文档?再次感谢 我主要提到了Javadocs。这就是它所说的:“设置当前别名配置。别名列表应采用“/aliasPath1=docBase1,/aliasPath2=docBase2”形式,其中 aliasPathN 必须包含前导 '/' 并且 docBaseN 必须是.war 文件或目录。”【参考方案3】:

这是我根据不同操作系统动态设置 Tomcat7 上下文别名的工作代码。当然你可以改进它

public class ContextListener implements ServletContextListener 
@Override
public void contextInitialized(ServletContextEvent sce) 
    ServletContext context = sce.getServletContext();

    // tomcat 7.x
    try 
        MBeanServer mBeanServer = MBeanServerFactory.findMBeanServer(null).get(0);
        ObjectName name = new ObjectName("Catalina", "type", "Server");
        Object server = mBeanServer.getAttribute(name, "managedResource");

        Object service = server.getClass().getMethod("findService", String.class).invoke(server, "Catalina");  //StandardService[Catalina]
        Object connectors = service.getClass().getMethod("findConnectors").invoke(service);
        Object engine = service.getClass().getMethod("getContainer").invoke(service);  //StandardEngine[Catalina]
        Object host = Array.get(engine.getClass().getMethod("findChildren").invoke(engine), 0);  //StandardHost[Catalina]
        Object stdContext = Array.get(host.getClass().getMethod("findChildren").invoke(host), 0);  //StandardContext[Catalina]
        Object mapper = stdContext.getClass().getMethod("getMapper").invoke(stdContext);
        //just a clean up step(remove the host)
        Field f1 = mapper.getClass().getDeclaredField("context");
        f1.setAccessible(true);
        Object ct = f1.get(mapper);

        Field f2 = ct.getClass().getDeclaredField("resources");
        f2.setAccessible(true);
        Object rs = f2.get(ct);

        Field f3 = rs.getClass().getDeclaredField("dirContext");
        f3.setAccessible(true);
        Object dc = f3.get(rs);

        mapper.getClass().getMethod("removeHost",String.class).invoke(mapper, host.getClass().getMethod("getName").invoke(host));
        //add the host back with all required aliases
        switch (OsCheck.getOperatingSystemType()) 
            case Windows:
                dc.getClass().getMethod("setAliases",String.class).invoke(dc,"/img/avatars=" + winAvatarAlias);
                break;
            default:
                dc.getClass().getMethod("setAliases",String.class).invoke(dc,"/img/avatars=" + linuxAvatarAlias);
                break;
        
        String ports = "";
        for (Object o :(Object[]) connectors ) 
            ports = ports + (Integer)o.getClass().getMethod("getPort").invoke(o) + " ";
        
        log.info("Tomcat 7.x detected, service , engine , host , stdContext , server port: ",
                service.getClass().getMethod("getName").invoke(service),
                engine.getClass().getMethod("getName").invoke(engine),
                host.getClass().getMethod("getName").invoke(host),
                stdContext.getClass().getMethod("getDisplayName").invoke(stdContext),
                ports);
     catch (Exception e) 
        e.printStackTrace();
    

【讨论】:

【参考方案4】:

基于 Khanh 的方法,这是一个适用于嵌入式 maven tomcat (v.7.0.62) 的上下文侦听器。

请注意差异(“Tomcat”而不是“Catalina”并且没有 findService("Catalina")),以便该方法适用于嵌入式 tomcat。与 Khanh 相比,我使用常规方法而不是反射来获取 BaseDirContext 对象。

最后,您应该注意,您需要在 BaseDirContext 对象而不是 StandardContext 对象上调用 setAliases()!在内部,StandardContext 的 setAliases() 只是一个 setter,而 BaseDirContext 的 setAliases() 做了很多其他的事情,所以已经运行的 tomcat 确实注册了你的新别名。

import org.apache.catalina.Container;
import org.apache.catalina.Server;
import org.apache.catalina.Service;
import org.apache.catalina.core.StandardContext;
import org.apache.catalina.core.StandardEngine;
import org.apache.log4j.Logger;
import org.apache.naming.resources.BaseDirContext;
import org.apache.naming.resources.ProxyDirContext;

import javax.management.MBeanServer;
import javax.management.MBeanServerFactory;
import javax.management.ObjectName;
import javax.servlet.ServletContext;
import javax.servlet.ServletContextEvent;
import javax.servlet.ServletContextListener;

public class AliasesContextListener implements ServletContextListener 
    private static Logger log = Logger.getLogger(AliasesContextListener.class);

    @Override
    public void contextInitialized(ServletContextEvent sce) 
        try 
            String aliases = "/foo=C:\\bar";

            //get current tomcat server, engine and context objects
            MBeanServer mBeanServer = MBeanServerFactory.findMBeanServer(null).get(0);
            ObjectName name = new ObjectName("Tomcat", "type", "Server");
            Server server = (Server) mBeanServer.getAttribute(name, "managedResource");
            Service[] services = server.findServices();
            StandardEngine engine = (StandardEngine) services[0].getContainer();
            Container defaultHostContainer = engine.findChild(engine.getDefaultHost());

            ServletContext servletContext = sce.getServletContext();
            StandardContext standardContext = (StandardContext) defaultHostContainer.findChild(servletContext.getContextPath());
            ProxyDirContext proxyDirContext = (ProxyDirContext) standardContext.getResources();
            BaseDirContext baseDirContext = (BaseDirContext) proxyDirContext.getDirContext();

            //modify the aliases entry
            baseDirContext.setAliases(aliases);
         catch (Exception e) 
            log.error("error while setting aliases in context listener", e);
        
    

    @Override
    public void contextDestroyed(ServletContextEvent sce) 
        //not implemented
    

【讨论】:

以上是关于如何在 java 中为 Servlet 上下文添加别名?的主要内容,如果未能解决你的问题,请参考以下文章

java web app-为servlet上下文侦听器向web.xml添加内容后无法看到jsp文件

如何在 cocos2d 中为图层添加滚动?

如何在eclipse中为java代码添加快捷键

如何在 tomcat 容器中安装 servlet 并将其加载到每个 Web 应用程序的上下文中?

如何刷新我在 ServletContextListener 中设置的 servlet 上下文变量?

如何在 swift 中为 UIPickerview 的 2 组件数组添加新元素