springMVC之放行静态资源、文件上传、拦截器

本文详细解析SpringMVC中静态资源处理及文件上传的两种方式,包括配置方法、示例代码及常见问题解决方案,适用于初学者及开发者快速上手。

放行静态资源

为什么静态资源无法访问

默认Tomcat是可以处理静态资源的(通过DefaultServlet处理在%tomcat_home%/conf/web.xml中),但是我们配置了SpringMVC后, 往往会将DispatcherServlet的拦截路径设置成"/",会将Tomcat中默认的DefaultServlet覆盖掉,所以静态资源也会进入DispatcherServlet中处理,而默认DispatcherServlet中是没有处理静态资源相关的逻辑, 所以访问静态资源就会报404

怎么解决

第一种方法

配置< mvc:default-servlet-handler />

将静态资源交还给容器来处理(比如说Tomcat就是让DefaultServlet来处理)

  • 优点:配置简单
  • 缺点:只能访问web根目录下公开的静态文件, WEB-INF、类路径下都无法访问

第二种方法

在dispatcher-servlet.xml配置文件中配置下面的一行配置即可

<!-- 放行静态资源, mapping匹配你请求的URI; location指定静态资源的位置(建议前后面都加/) -->
<mvc:resources mapping="/static/**" location="/static/" />

放行静态资源, mapping匹配你请求的URI; location指定静态资源相对位置,会将mapping中通配部分的内容拼到location指定的相对位置后面

示例

在这里插入图片描述
dispatcher-servlet.xml

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:mvc="http://www.springframework.org/schema/mvc"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xsi:schemaLocation="
        http://www.springframework.org/schema/beans
        http://www.springframework.org/schema/beans/spring-beans.xsd
        http://www.springframework.org/schema/mvc
        http://www.springframework.org/schema/mvc/spring-mvc.xsd">
        
    <!-- 放行静态资源, mapping匹配你请求的URI;location指定静态资源的位置(建议前后面都加/) -->
    <mvc:resources mapping="/static/**" location="/static/" />
</beans>

浏览器访问地址

http://localhost:8080/springmvc/static/index.html

文件上传

SpringMVC提供了一个MultipartResolver接口用于支持文件上传。 MultipartResolver有两个实现类分别基于apache file-upload包和Serlvet3.0 API方式的文件上传。他们是:

  • CommonsMultipartResolver (2.5)
  • StandardServletMultipartResolver (3.0)

2.5方式文件上传配置如下:

通过apache common file-upload包的Servlet 2.5的方式上传文件

  1. 在pom.xml中添加common-fileupload依赖
<dependency>
	<groupId>commons-fileupload</groupId>
	<artifactId>commons-fileupload</artifactId>
	<version>1.3.3</version>
</dependency>
  1. 在bean中配置CommonsMultipartResolver,最好以multipartResolver命名
<bean id="multipartResolver" class="org.springframework.web.multipart.commons.CommonsMultipartResolver" />

支持"maxUploadSize", “maxInMemorySize” and "defaultEncoding"三个参数

注意: apache file-upload相关的依赖需要自己另外添加进来

示例

index.jsp

<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
    <title>servlet 2.5方式文件上传</title>
</head>
<body>
    <form enctype="multipart/form-data" action="upload2" method="post">
        <input type="file" name="myFile" />
        <input type="submit" value="提交" />
    </form>
</body>
</html>

UploadController .java

@Controller
public class UploadController {
    @ResponseBody
    @RequestMapping("/upload2")
    public String upload2(@RequestParam("myFile") MultipartFile file){
        try {
            file.transferTo(new File("D:/" + file.getOriginalFilename()));
            return "上传成功";
        } catch (IOException e) {
            e.printStackTrace();
            return "上传失败";
        }
    }
}

3.0方式文件上传配置如下:

通过Servlet 3.0的方式上传文件

  1. 在web.xml 标签中配置: 启用DispatcherServlet的文件上传功能
<servlet>
	<servlet-name>dispatcher</servlet-name>
	<servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
	<!-- 指定SpringMVC 配置文件位置,DispatcherServlet初始化时会初始化Spring上下文(WebApplicationContext) -->
	<!-- 默认配置文件寻找位置:/WEB-INF/{servlet-name}-servlet.xml,如果名字符合默认寻找规则,可以不指定配置文件路径 -->
	<init-param>
		<param-name>contextConfigLocation</param-name>
		<param-value>WEB-INF/dispatcher-servlet.xml</param-value>
	</init-param>
	<!-- 配置容器启动时初始化DispatcherServlet -->
	<load-on-startup>1</load-on-startup>
	<!-- 启用DispatcherServlet的文件上传功能 -->
	<multipart-config />
</servlet>
<servlet-mapping>
	<servlet-name>dispatcher</servlet-name>
	<url-pattern>/</url-pattern>
</servlet-mapping>
  1. 在bean中配置StandardServletMultipartResolver,最好以multipartResolver命名
<bean name="multipartResolver" class="org.springframework.web.multipart.support.StandardServletMultipartResolver"></bean>

在Controller中处理文件上传:

@Controller
public class FileUploadController {
    @PostMapping("/form")
    public String handleFormUpload(@RequestParam("name") String name,@RequestParam("file")MultipartFile file) {
        if (!file.isEmpty()) {
            byte[] bytes = file.getBytes();
            // 将字节存储在某处
            return "redirect:uploadSuccess";
        }
        return "redirect:uploadFailure";
    }
}

示例

index.jsp

<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
    <title>servlet 2.5方式文件上传</title>
</head>
<body>
    <form enctype="multipart/form-data" action="upload3" method="post">
        <input type="file" name="myFile" />
        <input type="submit" value="提交" />
    </form>
</body>
</html>

UploadController .java

@Controller
public class UploadController {
    @ResponseBody
    @RequestMapping("/upload3")
    public String upload3(@RequestParam("myFile") MultipartFile file){
        try {
            file.transferTo(new File("D:/" + file.getOriginalFilename()));
            return "上传成功";
        } catch (IOException e) {
            e.printStackTrace();
            return "上传失败";
        }
    }
}

注意:上传文件响应中文内容会乱码
解决方法点击此处

拦截器

配置方法

在src\main\webapp\WEB-INF\dispatcher-servlet.xml配置文件中添加下面的配置

<!-- 配置拦截器 -->
<mvc:interceptors>
    <!-- 通过<bean>的形式添加interceptor,默认拦截所有请求 -->
    <bean class="com.lanou3g.springmvc.interceptor.TestInterceptor"/>
    <!-- 通过<mvc:interceptor>配置,可以进一步拦截哪些路径和不拦截哪些路径 -->
    <!--<mvc:interceptor>
        <mvc:mapping path="/*"/>
        <mvc:exclude-mapping path="upload2"/>
        <bean class="com.lanou3g.springmvc.interceptor.LoginInterceptor"/>
    </mvc:interceptor>-->
</mvc:interceptors>

拦截时机

(执行handler之前)preHandle
(执行handler之后)postHandle
(请求完成,返回到客户端之前)afterCompletion

示例

TestInterceptor.java

public class TestInterceptor implements HandlerInterceptor {

    /**
     *  在调用实际的handler执行之前被DispatcherServlet(大总管)调用
     * @param request
     * @param response
     * @param handler
     * @return
     * @throws Exception
     */
    @Override
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
        System.out.println("TestInterceptor-preHandle");
        return true;
    }

    /**
     * 在调用实际的handler执行之后被DispatcherServlet(大总管)调用
     * @param request
     * @param response
     * @param handler
     * @param modelAndView
     * @throws Exception
     */
    @Override
    public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {
        System.out.println("TestInterceptor-postHandle");
    }

    /**
     * 请求完成以后被DispatcherServlet(大总管)调用
     * @param request
     * @param response
     * @param handler
     * @param ex
     * @throws Exception
     */
    @Override
    public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
        System.out.println("TestInterceptor-afterCompletion");
    }
}
<think>嗯,用户问的是SpringMVC返回静态资源的介绍。我需要先回忆一下SpringMVC处理静态资源的基本机制。记得在SpringMVC中,默认情况下,DispatcherServlet会处理所有的请求,包括静态资源,比如图片、CSS、JS文件。但这样的话,可能会导致静态资源无法正确加载,因为DispatcherServlet没有找到对应的处理器。 那怎么解决这个问题呢?通常有三种方法对吧。第一种是配置默认Servlet处理,也就是在web.xml里把DefaultServlet的映射加上。不过这种方法可能依赖具体的容器,比如Tomcat的DefaultServlet。第二种方法是在Spring配置文件中使用<mvc:resources/>标签,指定静态资源的目录和路径。第三种是通过<mvc:default-servlet-handler/>,让容器默认的Servlet来处理静态资源,同时需要结合注解驱动。 需要详细说明每种方法的配置步骤和注意事项。比如用<mvc:resources/>的时候,要指定location和mapping,可能需要处理缓存或者版本控制。而使用默认Servlet处理的话,可能会影响其他动态请求的处理顺序,需要确保配置正确。 另外,用户可能想知道为什么默认情况下静态资源会被拦截,以及这些配置背后的原理。比如,当添加<mvc:resources/>时,Spring会注册ResourceHttpRequestHandler来处理静态资源,而<mvc:default-servlet-handler/>则是通过SimpleUrlHandlerMapping将请求转发给容器的默认Servlet。 还有可能用户在实际开发中遇到问题,比如配置了<mvc:resources/>但静态资源还是404,这时候需要检查路径是否正确,或者是否有其他拦截器干扰。同时,缓存策略的设置也很重要,避免浏览器缓存导致更新不及时。 需要提醒用户注意配置的顺序和位置,确保这些配置在Spring的上下文中正确加载。另外,在Spring Boot中,自动配置已经处理了静态资源,放在classpath下的static、public等目录即可,不需要手动配置,但用户可能是在用传统的Spring MVC项目,所以需要手动处理。 总结下来,回答的结构应该是先介绍问题,再分点讲解解决方法,包括配置方式和原理,最后可能补充常见问题和注意事项。确保用中文,步骤清晰,例子具体,比如给出XML配置的代码示例,以及可能涉及的路径写法。</think>SpringMVC 返回静态资源的实现机制及配置方法如下,按步骤解析: 1. **问题背景** - 默认情况下,SpringMVC 的 `DispatcherServlet` 会拦截所有请求(包括 `/*` 路径) - 导致静态资源(CSS/JS/图片)被误认为 Controller 请求,返回 404 错误 2. **解决方案** █ 方法1:配置默认 Servlet 处理 ```xml <!-- web.xml --> <servlet-mapping> <servlet-name>default</servlet-name> <url-pattern>*.js</url-pattern> <url-pattern>*.css</url-pattern> <url-pattern>/img/*</url-pattern> </servlet-mapping> ``` *特点*:依赖容器实现,需明确资源后缀 █ 方法2:使用 `<mvc:resources/>` 标签 ```xml <!-- springmvc-config.xml --> <mvc:resources mapping="/static/**" location="/resources/" cache-period="31556926"/> ``` *参数说明*: - `mapping`:虚拟路径(支持 Ant 风格) - `location`:物理路径(可多个,用逗号分隔) - `cache-period`:缓存时间(秒) █ 方法3:启用默认 Servlet 转发 ```xml <mvc:default-servlet-handler/> <mvc:annotation-driven/> ``` *执行顺序*: ```mermaid graph LR A[请求] --> B{是否有对应Handler?} B -->|是| C[Controller处理] B -->|否| D[DefaultServlet处理] ``` 3. **技术原理** - 资源处理器 `ResourceHttpRequestHandler` 负责处理静态文件 - 支持版本控制(通过 `ResourceUrlProvider`) - MIME 类型自动识别(依赖 `javax.activation` 包) 4. **Spring Boot 优化方案** ```properties # application.properties spring.mvc.static-path-pattern=/content/** spring.web.resources.static-locations=classpath:/files/ ``` *默认搜索路径*: - `classpath:/META-INF/resources/` - `classpath:/resources/` - `classpath:/static/` - `classpath:/public/` 5. **常见问题排查** - 404 错误:检查路径是否包含中文/特殊字符 - 缓存问题:添加版本号 `/static/style.css?v=1.0` - 安全拦截:在 Spring Security 配置中放行静态资源 ```java http.authorizeRequests() .antMatchers("/static/**").permitAll(); ``` 建议根据项目需求选择方案:传统项目推荐方法2,Spring Boot 项目使用自动配置即可。注意路径映射与实际文件位置的对应关系,可通过浏览器开发者工具 Network 面板验证请求响应状态。
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值