[细节]Tomcat对静态资源的缓存支持

本文深入探讨了Tomcat如何通过DefaultServlet支持静态资源的访问,并分析了其内置的缓存功能。从配置DefaultServlet到解析关键方法,揭示了控制Tomcat静态资源缓存行为的方法。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

使用tomcat时我们可以通过配置DefaultServlet来支持对静态资源的访问,而DefaultServlet是具有缓存功能的,下面通过对关键的源码对其进行分析并介绍如何通过配置来控制tomcat静态资源的缓存行为。

1. 配置DefaultServlet:

我们有两种方式来配置资源,一种是通过部署描述符,另一种就是通过代码进行配置:

通过代码配置DefaultServlet:
	<servlet>  
	  <servlet-name>default</servlet-name>  
	  <servlet-class>org.apache.catalina.servlets.DefaultServlet</servlet-class>  
	  <init-param>  
		  <param-name>debug</param-name>  
		  <param-value>0</param-value>  
	  </init-param>  
	  <init-param>  
		  <param-name>listings</param-name>  
		  <param-value>false</param-value>  
	  </init-param>  
	  <load-on-startup>1</load-on-startup>  
	</servlet> 

	<servlet-mapping>
     	<servlet-name>default</servlet-name>
    	 <url-pattern>*.css</url-pattern>
	</servlet-mapping>
	 
	<servlet-mapping>
	    <servlet-name>default</servlet-name>
	    <url-pattern>*.gif</url-pattern>
	 </servlet-mapping>
	    
	 <servlet-mapping>
	     <servlet-name>default</servlet-name>
	     <url-pattern>*.jpg</url-pattern>
	 </servlet-mapping>
	    
	 <servlet-mapping>
	     <servlet-name>default</servlet-name>
	     <url-pattern>*.js</url-pattern>
	 </servlet-mapping>
	 
	 <servlet-mapping>
	     <servlet-name>default</servlet-name>
	     <url-pattern>*.swf</url-pattern>
	 </servlet-mapping>

通过代码配置:
container.getServletRegistration("default").addMapping("/resource/*");
这句代码在ServletContainerInitializer(Servlet 3.0之后)的onStartUp方法中,这个事件点是应用程序程序开始我们能到达的最早点,在ServetContextListener的contextInitialized方法之前。因为DefaultServlet实例tomcat会默认创建,因此我们可以通过上面的语句直接映射到指定的静态资源路径。

2. DefaultServlet的关键方法:

既然是HttpServlet的子类,我们首先看看doGet方法:
    @Override
    protected void doGet(HttpServletRequest request,
                         HttpServletResponse response)
        throws IOException, ServletException {

        // Serve the requested resource, including the data content
        serveResource(request, response, true, fileEncoding);

    }
下面是serveResource方法的一些关键逻辑:

根据路径获取资源对象:
<span style="white-space:pre">	</span>WebResource resource = resources.getResource(path);
当资源是文件时,检查相关头部,其中checkIfHeaders方法中就对Etag和lastModified头进行检查:
<span style="white-space:pre">	</span>if (resource.isFile()) {
            // Checking If headers
            included = (request.getAttribute(
                    RequestDispatcher.INCLUDE_CONTEXT_PATH) != null);
            if (!included && !isError && !checkIfHeaders(request, response, resource)) {
                return;
            }
        }

<span style="white-space:pre">	</span>protected boolean checkIfHeaders(HttpServletRequest request,
                                     HttpServletResponse response,
                                     WebResource resource)
        throws IOException {

        return checkIfMatch(request, response, resource)
            && checkIfModifiedSince(request, response, resource)
            && checkIfNoneMatch(request, response, resource)
            && checkIfUnmodifiedSince(request, response, resource);

    }
可以看到checkIfHeaders方法检查了4个头:If-Match、If-Modified-Since、If-None-Match、If-Unmodified-Since;
其中如果设置If-None-Match的话,If-Modified-Since会忽略,这两个头分别对应于Etag和Last-Modified,如果没有改变,返回304 Not Modified,对应的方法返回false;
而If-Match头和If-Unmodified-Since头分别表示Etag和last-modified值未改变应该返回文件数据(200 OK),对应的方法返回true,如果不相等返回状态码412,方法返回false;
PS:If-Modified-Since和If-None-Match方法会设置last-modified和Etag响应头;

由此可见,只要这个4个方法任何一个返回了false,都直接返回,不返回文件数据,有两种可能:304, 412;

如果上一步没有return,那么应当返回文件数据:
获取资源对象的eTag和lastModifiedHttp并设置响应的头;
<span style="white-space:pre">	</span>String eTag = null;
        String lastModifiedHttp = null;
        if (resource.isFile() && !isError) {
            eTag = resource.getETag();
            lastModifiedHttp = resource.getLastModifiedHttp();
        }

上述过程展示了在什么情况下返回:200, 304, 412;那么Tomcat是如何保存Etag和Last-Modified值的呢?关键在与resources.getResource(path);resources的类型是WebResourceRoot,在Tomcat中的实现是StandardContext,我们来看一下getResource方法:
<span style="white-space:pre">	</span>private WebResource getResource(String path, boolean validate,
            boolean useClassLoaderResources) {
        if (validate) {
            path = validate(path);
        }

        if (isCachingAllowed()) {
            return cache.getResource(path, useClassLoaderResources);
        } else {
            return getResourceInternal(path, useClassLoaderResources);
        }
    }
cachingAllowed的值默认为true,还有一个相关的值maxSize(Cache类的成员变量),它们在server.xml的<Context>节点中设置:
<span style="white-space:pre">	</span><Context
		crossContext="true"
		privileged="true"
		path=""
		docBase="/usr/local/example.war"
		reloadable="false"
		unpackWAR="true"
		<span style="color:#ff0000;">cachingAllowed</span>="true"
		<span style="color:#ff0000;">cacheMaxSize</span>="1024"
	></Context>
Cache类负责了实际的缓存工作:开启了缓存之后,会将Resource对象放到一个ConcurrentMap中:
private final ConcurrentMap<String,CachedResource> resourceCache =
            new ConcurrentHashMap<>();












<think>好的,我现在需要解决用户在Tomcat中遇到的静态资源返回404的问题。根据用户提供的引用内容,我需要先理解问题的根源,然后逐步分析可能的原因和解决方案。 首先,用户提到在web.xml中配置了url-pattern为/会导致所有请求都由DispatcherServlet处理,包括静态资源,这可能引起静态资源404的问题[^1]。这说明当使用Spring MVC时,如果DispatcherServlet处理了所有请求,但没有正确配置静态资源的处理,就会导致无法找到静态资源。 接下来,引用[2]中提到Tomcat资源访问404的可能原因,包括配置路由的方式(注解或XML)、XML文件错误、路径配置问题以及项目未正确部署到Tomcat[^2]。需要检查这些方面是否存在配置错误,比如web.xml中的servlet映射是否正确,或者Spring MVC的配置文件是否正确处理了静态资源。 然后,引用[3]和[4]虽然主要讨论中文乱码问题,但其中提到的路径配置和字符编码设置也可能间接影响资源访问,但用户当前的问题是404,所以重点应放在资源是否存在和路径配置上[^3][^4]。 接下来,我需要整理常见的原因及对应的解决方案: 1. **DispatcherServlet配置问题**:当url-pattern设置为/时,Spring MVC会拦截所有请求,包括静态资源。需要检查是否在Spring配置文件中启用静态资源的处理,例如通过<mvc:resources>标签或配置资源处理器。 2. **静态资源路径不正确**:静态资源可能没有放在正确的目录下,如WEB-INF目录外的资源无法直接访问,应放在webapp或resources目录下,并确保路径正确。 3. **Tomcat部署问题**:项目可能没有正确部署到Tomcat服务器,检查Tomcat的webapps目录下是否存在项目文件,或者IDEA中部署配置是否正确。 4. **XML配置错误**:检查web.xml或Spring配置文件中的路径配置是否有误,例如是否遗漏了必要的配置项,或者路径结尾缺少斜杠。 5. **缓存问题**:浏览器或Tomcat缓存可能导致资源未更新,尝试清除缓存或重启Tomcat。 每个原因对应的解决方案需要具体说明,例如如何配置<mvc:resources>,如何检查目录结构,如何验证项目部署等。同时,需要确保用户按照步骤操作后能够解决404问题。 最后,生成相关问题,帮助用户进一步排查可能的相关问题,如中文乱码或性能优化等。</think>### Tomcat静态资源返回404的原因及解决方案 #### 一、问题原因分析 1. **Spring MVC拦截了静态资源请求** 当`web.xml`中配置`url-pattern`为`/`时,所有请求(包括静态资源)会被`DispatcherServlet`处理。若未配置静态资源映射,Spring MVC会尝试通过控制器解析路径,导致404。 2. **静态资源路径配置错误** - 资源未放置在`webapp`或`resources/static`目录下(取决于项目结构)。 - 访问路径与实际文件路径不匹配,例如:文件在`webapp/images/logo.png`,但访问路径为`/static/logo.png`。 3. **Tomcat部署问题** - 项目未正确挂载到Tomcat的`webapps`目录[^2]。 - 开发工具(如IDEA)未将资源文件打包到部署目录。 4. **XML配置文件错误** - `web.xml`中`<servlet-mapping>`配置冲突。 - Spring MVC配置文件中未启用静态资源处理(如缺少`<mvc:resources>`)。 5. **缓存或未清理旧文件** Tomcat或浏览器缓存可能导致旧资源路径无法更新。 --- #### 二、逐步解决方案 ##### 1. 配置Spring MVC放行静态资源 在`springmvc.xml`中添加以下配置: ```xml <mvc:resources mapping="/static/**" location="/static/"/> ``` - `mapping`:定义URL路径模式(如`/static/css/style.css`)。 - `location`:指定实际资源目录(如`webapp/static/`)[^1]。 ##### 2. 检查静态资源路径 - **目录结构示例**: ``` project └── webapp ├── static │ ├── css │ └── images └── WEB-INF ``` - 访问时使用:`http://localhost:8080/project/static/images/logo.png`。 ##### 3. 验证Tomcat部署 - 检查Tomcat的`webapps`目录下是否存在项目文件夹。 - 确保开发工具(如Maven)已将资源打包到`target`目录。 ##### 4. 修复XML配置 - **web.xml示例**: ```xml <servlet> <servlet-name>dispatcher</servlet-name> <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class> </servlet> <servlet-mapping> <servlet-name>dispatcher</servlet-name> <url-pattern>/</url-pattern> </servlet-mapping> ``` - 确保`<url-pattern>`与其他Servlet无冲突。 ##### 5. 清理缓存并重启 - 清除浏览器缓存。 - 删除Tomcat的`work`目录(缓存目录)后重启服务。 --- #### 三、验证步骤 1. 直接通过URL访问静态资源(如`http://localhost:8080/static/test.txt`)。 2. 检查Tomcat日志是否有路径解析错误。 3. 使用开发者工具(浏览器F12)查看网络请求状态码。 ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值