我的问题很简单。我希望在/static/*
上下文中提供静态资源,但是我的特定servlet服务于/*
上下文。由于/static/*
是/*
的子集,因此不起作用。
我的web.xml看起来像这样:
<web-app version="2.5" xmlns="http://java.sun.com/xml/ns/javaee"xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd">
<display-name>template-guice-jersey-tomcat</display-name>
<session-config>
<session-timeout>30</session-timeout>
</session-config>
<welcome-file-list>
<welcome-file>index.jsp</welcome-file>
</welcome-file-list>
<!-- set up Google Guice Servlet integration -->
<filter>
<filter-name>guiceFilter</filter-name>
<filter-class>com.google.inject.servlet.GuiceFilter</filter-class>
</filter>
<filter-mapping>
<filter-name>guiceFilter</filter-name>
<url-pattern>/*</url-pattern>
</filter-mapping>
<servlet-mapping>
<servlet-name>default</servlet-name>
<url-pattern>/static/*</url-pattern>
</servlet-mapping>
<listener>
<listener-class>de.danbim.templateguicejerseytomcat.GuiceServletConfig</listener-class>
</listener>
</web-app>
答案 0 :(得分:5)
以下是解决此问题的非常简洁的解决方案:http://www.kuligowski.pl/java/rest-style-urls-and-url-mapping-for-static-content-apache-tomcat,5:
不幸的是,在查看DefaultServlet源代码后,我发现DefaultServlet只接受请求URL的pathInfo部分,因此如果您的请求是/static/styles.css,容器会将其转换为/styles.css。 DefaultServlet省略了Servlet部分。如果你想访问这样的css文件,你应该使用/static/static/styles.css请求网址。
我们问题的简单解决方案是编写DefaultFilter类并将其放在web.xml文件的开头。此过滤器将所有静态内容调用转发到DefaultServlet。
定义一个过滤器,将请求分派给默认的servlet:
public class DefaultFilter implements Filter {
private RequestDispatcher defaultRequestDispatcher;
@Override
public void destroy() {}
@Override
public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain)
throws IOException, ServletException {
defaultRequestDispatcher.forward(request, response);
}
@Override
public void init(FilterConfig filterConfig) throws ServletException {
this.defaultRequestDispatcher =
filterConfig.getServletContext().getNamedDispatcher("default");
}
}
将过滤器(在其他过滤器之前)添加到web.xml:
<filter>
<filter-name>default</filter-name>
<servlet-name>default</servlet-name>
<filter-class>pl.kuligowski.example.DefaultFilter</filter-class>
</filter>
<filter-mapping>
<filter-name>default</filter-name>
<url-pattern>/static/*</url-pattern>
<url-pattern>*.ico</url-pattern>
</filter-mapping>
只有静态内容调用与DefaultFilter匹配,后者只是破坏过滤器链并将请求转发到DefaultServlet。
答案 1 :(得分:1)
对我来说,这适用于Tomcat 8.5(我甚至没有web.xml):
@WebServlet(urlPatterns = "/*")
public class MainServlet {}
@WebServlet(urlPatterns = { "/static/*", "/gwt/*" })
public class StaticServlet extends org.apache.catalina.servlets.DefaultServlet {}
更具体的网址模式似乎自动获得优先权。要扩展catalina默认servlet,请将org.apache.tomcat > tomcat-catalina
maven依赖项和范围provided
添加到项目中。
你可能(像我一样)立即运行一个问题,即只有应用程序上下文的请求在解析html中的相对路径方面存在问题(例如,对于样式表和javascripts)。
我通过主servlet中的重定向解决了这个问题:
@Override
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
if(request.getPathInfo() == null) {
// redirect http://localhost:8080/test -> http://localhost:8080/test/ (add trailing slash)
// because only then are the relative paths to the gwt scripts correctly resolved
getResponse().sendRedirect(getRequest().getContextPath() + "/");
return;
}
//...
}