2024-01-17
原文作者:路人 原文地址: http://www.itsoku.com/course/6/222

1、本文内容

  • 静态资源 & jsp 是被谁处理的?
  • 如何让springmvc处理所有请求?
  • springmvc如何处理静态资源?
  • springmvc处理静态资源的2种方案、区别、原理详解
  • 领取后端必备的200本书籍

2、静态资源 & jsp 是被谁处理的?

不知大家是否注意了一个问题,我们将web项目发布到tomcat后,可以直接访问jsp后缀的文件,或者直接访问项目中的静态资源(js/css/图片之类的文件),只要路径写的没有问题,都可以正常在浏览器中显示,大家是否思考过,是谁在处理这个请求呢?

tomcat是一个servlet容器,servlet容器中通常有servlet来处理各种请求,那么当我们访问静态资源或者jsp的时候,当然也是由servlet来处理这些请求的,但是我们并没有在web.xml中配置由哪个servlet来处理这些请求啊,这是因为tomcat中自动帮我们配置了一些默认的servlet来处理这类的请求。

tomcat的安装目录中有个web.xml文件,位于(apache-tomcat-9.0.46\conf\web.xml),这个文件相当于对发布到tomcat中的所有web项目的默认配置,我们来看一下这个文件,有几个重要的servlet配置信息,这些servlet可以用来处理jsp后置的文件或者静态资源,抽取部分配置如下

    <servlet>
    	<servlet-name>default</servlet-name>
    	<servlet-class>org.apache.catalina.servlets.DefaultServlet</servlet-class>
    	<load-on-startup>1</load-on-startup>
    </servlet>
    
    <servlet>
    	<servlet-name>jsp</servlet-name>
    	<servlet-class>org.apache.jasper.servlet.JspServlet</servlet-class>
    	<load-on-startup>3</load-on-startup>
    </servlet>
    
    <servlet-mapping>
    	<servlet-name>default</servlet-name>
    	<url-pattern>/</url-pattern>
    </servlet-mapping>
    
    <servlet-mapping>
    	<servlet-name>jsp</servlet-name>
    	<url-pattern>*.jsp</url-pattern>
    	<url-pattern>*.jspx</url-pattern>
    </servlet-mapping>

配置中主要定义了2个servlet

  1. JspServlet用来处理jsp文件
  2. DefaultServlet会处理/开头的请求,这里重点说下这个servlet的处理逻辑,它会处理所有其他servlet不匹配的请求,当一个请求过来之后,会先去匹配其他的servlet,当找不到匹配的servlet的时候,才会走DefaultServlet
  3. 我们可以在项目中的web.xml中对tomcat/config/web.xml中的配置进行覆盖,即项目中的web.xml相对于tomcat中的web.xml有更高的优先级。

3、如何让springmvc处理所有请求?

如果我们想让springmvc来处理所有的请求,怎么做呢?只需修改web.xml中的配置,将springmvc这个servlet的url-pattern的值设置为/就可以了,此时所有的请求都会被springmvc处理,即所有的请求都会被DispatcherServlet处理。

202401172041283301.png

4、问题来了:静态资源404了

当我们将上面springmvc的url-pattern改为/时,大家可以自己去试试,此时再访问静态资源的时候,如下图,会报404,这个请求被DispatcherServlet 接收了,但是这个servlet默认情况下并没有处理静态资源的能力,即找不到请求的资源,所以才会报404

202401172041288682.png

springmvc提供了2种解决方案,来看下。

5、方案2:使用mvc:default-servlet-handler/

在springmvc的配置文件中添加下面配置

    <mvc:annotation-driven/>
    <mvc:default-servlet-handler/>

这里说一下原理

1、加入mvc:default-servlet-handler/配置后,框架会创建DefaultServletHttpRequestHandler对象,这个对象会将请转发给tomcat的default这个servlet处理,所以静态资源就可以正常访问了

2、注意:配置中的第1行<mvc:annotation-driven/>也必须加上,如果不加上,@RequestMapping的请求就无法处理了

3、这种用法有个前提是,web容器中需要有default这个servlet,且这个servlet可以处理静态资源

6、方案2:使用mvc:resources/

从spring3.0版本后,spring定义了专门处理静态资源访问请求的处理器ResourceHttpRequestHandler,并且添加了mvc:resources/标签,用来处理静态资源无法访问的问题。

需要在springmvc配置文件中添加如下配置

    <mvc:annotation-driven/>
    <mvc:resources mapping="/imgs/**" location="/static/imgs/"/>
    <mvc:resources mapping="/css/**" location="/static/css/"/>
    <mvc:resources mapping="/js/**" location="/static/js/"/>
  • mvc:resource 标签用来指定请求的静态资源的url和本地地址的映射关系
  • mapping:用来指定请求的地址的,比如/imgs/**:可以匹配任何以imgs开头的url,如:/imgs/1.jpg、/imgs/2.jpg、/imgs/order/3.jpg
  • location:用来指定静态资源所在项目中的目录
  • 注意:配置中的第1行<mvc:annotation-driven/>也必须加上,如果不加上,@RequestMapping的请求就无法处理了
  • 这种用法不依赖于web容器,所以比第一种方案要好一些,也建议大家用这种方案。

上面代码中配置了3行mvc:resources,可以优化下,如下

    <mvc:annotation-driven/>
    <mvc:resources mapping="/static/**" location="/static/"/>

7、总结

  • tomcat默认能够处理jsp和静态资源,是因为tomcat中默认有2个servlet:default和jsp
  • 到此我们已经知道如何让springmvc来处理所有请求了,只需要在web.xml中调整一下url-pattern的值为/就可以了
  • 默认情况下,springmvc无法处理静态资源,本文介绍了2种方式,大家可以回头在再去看看,第一种需要依赖于web容器中的default这个servlet,使用上面对web容器有依赖作用;而第2种方式是springmvc推荐的方式,更好一些,对web容器没有依赖
  • 不管用那种方式,都不要忘记了<mvc:annotation-driven/>这个注解,否则咱们自定义的controller中的@RequestMapping就无效了

8、案例代码

    git地址:https://gitee.com/javacode2018/springmvc-series
阅读全文