[Java] SpringMVC工作原理之四:MultipartResolver
MultipartResolver 用于处理文件上传,当收到请求时 DispatcherServlet 的 checkMultipart() 方法会调用 MultipartResolver 的 isMultipart() 方法判断请求中是否包含文件。如果请求数据中包含文件,则调用 MultipartResolver 的 resolveMultipart() 方法对请求的数据进行解析,然后将文件数据解析成 MultipartFile 并封装在 MultipartHttpServletRequest (继承了 HttpServletRequest) 对象中,最后传递给 Controller,在 MultipartResolver 接口中有如下方法:
- boolean isMultipart(HttpServletRequest request); // 是否是 multipart
- MultipartHttpServletRequest resolveMultipart(HttpServletRequest request); // 解析请求
- void cleanupMultipart(MultipartHttpServletRequest request);
MultipartFile 封装了请求数据中的文件,此时这个文件存储在内存中或临时的磁盘文件中,需要将其转存到一个合适的位置,因为请求结束后临时存储将被清空。在 MultipartFile 接口中有如下方法:
- String getName(); // 获取参数的名称
- String getOriginalFilename(); // 获取文件的原名称
- String getContentType(); // 文件内容的类型
- boolean isEmpty(); // 文件是否为空
- long getSize(); // 文件大小
- byte[] getBytes(); // 将文件内容以字节数组的形式返回
- InputStream getInputStream(); // 将文件内容以输入流的形式返回
- void transferTo(File dest); // 将文件内容传输到指定文件中
MultipartResolver 是一个接口,它的实现类如下图所示,分为 CommonsMultipartResolver 类和 StandardServletMultipartResolver 类。
其中 CommonsMultipartResolver 使用 commons Fileupload 来处理 multipart 请求,所以在使用时,必须要引入相应的 jar 包;而 StandardServletMultipartResolver 是基于 Servlet 3.0来处理 multipart 请求的,所以不需要引用其他 jar 包,但是必须使用支持 Servlet 3.0的容器才可以,以tomcat为例,从 Tomcat 7.0.x的版本开始就支持 Servlet 3.0了。
一、CommonsMultipartResolver
1 使用方式
1.1 配置文件

- <!-- 定义文件上传解析器 -->
- <bean id="multipartResolver" class="org.springframework.web.multipart.commons.CommonsMultipartResolver">
- <!-- 设定默认编码 -->
- <property name="defaultEncoding" value="UTF-8"></property>
- <!-- 设定文件上传的最大值为5MB,5*1024*1024 -->
- <property name="maxUploadSize" value="5242880"></property>
- <!-- 设定文件上传时写入内存的最大值,如果小于这个参数不会生成临时文件,默认为10240 -->
- <property name="maxInMemorySize" value="40960"></property>
- <!-- 上传文件的临时路径 -->
- <property name="uploadTempDir" value="fileUpload/temp"></property>
- <!-- 延迟文件解析 -->
- <property name="resolveLazily" value="true"/>
- </bean>

1.2 上传表单
要在 form 标签中加入 enctype="multipart/form-data" 表示该表单要提交文件。
- <form action="${pageContext.request.contextPath}/test/file-upload.do" method="post" enctype="multipart/form-data">
- <input type="file" name="file">
- <input type="submit" value="提交">
- </form>
1.3 处理文件

- @RequestMapping("/file-upload")
- public ModelAndView upload(@RequestParam(value = "file", required = false) MultipartFile file,
- HttpServletRequest request, HttpSession session) {
- // 文件不为空
- if(!file.isEmpty()) {
- // 文件存放路径
- String path = request.getServletContext().getRealPath("/");
- // 文件名称
- String name = String.valueOf(new Date().getTime()+"_"+file.getOriginalFilename());
- File destFile = new File(path,name);
- // 转存文件
- try {
- file.transferTo(destFile);
- } catch (IllegalStateException | IOException e) {
- e.printStackTrace();
- }
- // 访问的url
- String url = request.getScheme() + "://" + request.getServerName() + ":" + request.getServerPort()
- + request.getContextPath() + "/" + name;
- }
- ModelAndView mv = new ModelAndView();
- mv.setViewName("other/home");
- return mv;
- }

2 源码分析
CommonsMultipartResolver 实现了 MultipartResolver 接口,resolveMultipart() 方法如下所示,其中 resolveLazily 是判断是否要延迟解析文件(通过XML可以设置)。当 resolveLazily 为 flase 时,会立即调用 parseRequest() 方法对请求数据进行解析,然后将解析结果封装到 DefaultMultipartHttpServletRequest 中;而当 resolveLazily 为 true 时,会在 DefaultMultipartHttpServletRequest 的 initializeMultipart() 方法调用 parseRequest() 方法对请求数据进行解析,而 initializeMultipart() 方法又是被 getMultipartFiles() 方法调用,即当需要获取文件信息时才会去解析请求数据,这种方式用了懒加载的思想。

- @Override
- public MultipartHttpServletRequest resolveMultipart(final HttpServletRequest request) throws MultipartException {
- Assert.notNull(request, "Request must not be null");
- if (this.resolveLazily) {
- //懒加载,当调用DefaultMultipartHttpServletRequest的getMultipartFiles()方法时才解析请求数据
- return new DefaultMultipartHttpServletRequest(request) {
- @Override //当getMultipartFiles()方法被调用时,如果还未解析请求数据,则调用initializeMultipart()方法进行解析 protected void initializeMultipart() {
- MultipartParsingResult parsingResult = parseRequest(request);
- setMultipartFiles(parsingResult.getMultipartFiles());
- setMultipartParameters(parsingResult.getMultipartParameters());
- setMultipartParameterContentTypes(parsingResult.getMultipartParameterContentTypes());
- }
- };
- } else {
- //立即解析请求数据,并将解析结果封装到DefaultMultipartHttpServletRequest对象中
- MultipartParsingResult parsingResult = parseRequest(request);
- return new DefaultMultipartHttpServletRequest(request, parsingResult.getMultipartFiles(),
- parsingResult.getMultipartParameters(), parsingResult.getMultipartParameterContentTypes());
- }
- }

在上面的代码中可以看到,对请求数据的解析工作是在 parseRequest() 方法中进行的,继续看一下 parseRequest() 方法源码

- protected MultipartParsingResult parseRequest(HttpServletRequest request) throws MultipartException {
- // 获取请求的编码类型
- String encoding = determineEncoding(request);
- FileUpload fileUpload = prepareFileUpload(encoding);
- try {
- List<FileItem> fileItems = ((ServletFileUpload) fileUpload).parseRequest(request);
- return parseFileItems(fileItems, encoding);
- } catch (...) {}
- }

在 parseRequest() 方法中,首先调用了 prepareFileUpload() 方法来根据编码类型确定一个 FileUpload 实例,然后利用这个 FileUpload 实例解析请求数据后得到文件信息,最后将文件信息解析成 CommonsMultipartFile (实现了 MultipartFile 接口) 并包装在 MultipartParsingResult 对象中。
二、StandardServletMultipartResolver
1 使用方式
1.1 配置文件
- <bean id="multipartResolver" class="org.springframework.web.multipart.support.StandardServletMultipartResolver">
- </bean>
这里并没有配置文件大小等参数,这些参数的配置在 web.xml 中

- <servlet>
- <servlet-name>springmvc</servlet-name>
- <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
- <init-param>
- <param-name>contextConfigLocation</param-name>
- <param-value>classpath:springmvc.xml</param-value>
- </init-param>
- <load-on-startup>1</load-on-startup>
- <multipart-config>
- <!-- 临时文件的目录 -->
- <location>d:/</location>
- <!-- 上传文件最大2M -->
- <max-file-size>2097152</max-file-size>
- <!-- 上传文件整个请求不超过4M -->
- <max-request-size>4194304</max-request-size>
- </multipart-config>
- </servlet>

1.2 上传表单
要在 form 标签中加入 enctype="multipart/form-data" 表示该表单要提交文件。
- <form action="${pageContext.request.contextPath}/test/file-upload.do" method="post" enctype="multipart/form-data">
- <input type="file" name="file">
- <input type="submit" value="提交">
- </form>
1.3 处理文件
1.3.1 通过 MultipartFile 类型的参数

- @RequestMapping("/file-upload")
- public ModelAndView upload(@RequestParam(value = "file", required = false) MultipartFile file,
- HttpServletRequest request, HttpSession session) {
- // 文件不为空
- if(!file.isEmpty()) {
- // 文件存放路径
- String path = request.getServletContext().getRealPath("/");
- // 文件名称
- String name = String.valueOf(new Date().getTime()+"_"+file.getOriginalFilename());
- File destFile = new File(path,name);
- // 转存文件
- try {
- file.transferTo(destFile);
- } catch (IllegalStateException | IOException e) {
- e.printStackTrace();
- }
- // 访问的url
- String url = request.getScheme() + "://" + request.getServerName() + ":" + request.getServerPort()
- + request.getContextPath() + "/" + name;
- }
- ModelAndView mv = new ModelAndView();
- mv.setViewName("other/home");
- return mv;
- }

1.3.2 通过 MultipartHttpServletRequest 类型的参数

- @RequestMapping("/file-upload")
- public ModelAndView upload(MultipartHttpServletRequest request, HttpSession session) {
- // 根据页面input标签的name
- MultipartFile file = request.getFile("file");
- // 文件不为空
- if(!file.isEmpty()) {
- // 文件存放路径
- String path = request.getServletContext().getRealPath("/");
- // 文件名称
- String name = String.valueOf(new Date().getTime()+"_"+file.getOriginalFilename());
- File destFile = new File(path,name);
- // 转存文件
- try {
- file.transferTo(destFile);
- } catch (IllegalStateException | IOException e) {
- e.printStackTrace();
- }
- // 访问的url
- String url = request.getScheme() + "://" + request.getServerName() + ":" + request.getServerPort()
- + request.getContextPath() + "/" + name;
- }
- ModelAndView mv = new ModelAndView();
- mv.setViewName("other/home");
- return mv;
- }

2 源码分析
StandardServletMultipartResolver 实现了 MultipartResolver 接口,resolveMultipart() 方法如下所示,其中 resolveLazily 是判断是否要延迟解析文件(通过XML可以设置)。

- public MultipartHttpServletRequest resolveMultipart(HttpServletRequest request) throws MultipartException {
- return new StandardMultipartHttpServletRequest(request, this.resolveLazily);
- }
- public StandardMultipartHttpServletRequest(HttpServletRequest request, boolean lazyParsing) throws MultipartException {
- super(request);
- // 判断是否立即解析
- if (!lazyParsing) {
- parseRequest(request);
- }
- }

对请求数据的解析工作是在 parseRequest() 方法中进行的,继续看一下 parseRequest() 方法源码

- private void parseRequest(HttpServletRequest request) {
- try {
- Collection<Part> parts = request.getParts();
- this.multipartParameterNames = new LinkedHashSet<String>(parts.size());
- MultiValueMap<String, MultipartFile> files = new LinkedMultiValueMap<String, MultipartFile>(parts.size());
- for (Part part : parts) {
- String disposition = part.getHeader(CONTENT_DISPOSITION);
- String filename = extractFilename(disposition);
- if (filename == null) {
- filename = extractFilenameWithCharset(disposition);
- }
- if (filename != null) {
- files.add(part.getName(), new StandardMultipartFile(part, filename));
- } else {
- this.multipartParameterNames.add(part.getName());
- }
- }
- setMultipartFiles(files);
- } catch (Throwable ex) {}
- }

parseRequest() 方法利用了 servlet3.0 的 request.getParts() 方法获取上传文件,并将其封装到 MultipartFile 对象中。
[Java] SpringMVC工作原理之四:MultipartResolver的更多相关文章
- [Java] SpringMVC工作原理之一:DispatcherServlet
一.DispatcherServlet 处理流程 在整个 Spring MVC 框架中,DispatcherServlet 处于核心位置,它负责协调和组织不同组件完成请求处理并返回响应工作.在看 Di ...
- [Java] SpringMVC工作原理之二:HandlerMapping和HandlerAdapter
一.HandlerMapping 作用是根据当前请求的找到对应的 Handler,并将 Handler(执行程序)与一堆 HandlerInterceptor(拦截器)封装到 HandlerExecu ...
- [Java] SpringMVC工作原理之三:ViewResolver
一.ViewResolver 根据视图的名称将其解析为 View 类型的视图,如通过 ModelAndView 中的视图名称将其解析成 View,View 是用来渲染页面的,也就是将 Model 填入 ...
- SpringMVC工作原理详解
先来看一下什么是 MVC 模式 MVC 是一种设计模式. MVC 的原理图如下: SpringMVC 简单介绍 SpringMVC 框架是以请求为驱动,围绕 Servlet 设计,将请求发给控制器,然 ...
- SpringMVC 工作原理详解
本文Github开源项目https://github.com/Snailclimb/JavaGuide,只供自己学习总结无商业用途,如有侵权,联系删除 先来看一下什么是 MVC 模式 MVC 是一种设 ...
- Java虚拟机工作原理详解 (一)
一.类加载器 首先来看一下java程序的执行过程. 从这个框图很容易大体上了解java程序工作原理.首先,你写好java代码,保存到硬盘当中.然后你在命令行中输入 javac YourClassNam ...
- Java虚拟机工作原理详解
原文地址:http://blog.csdn.net/bingduanlbd/article/details/8363734 一.类加载器 首先来看一下java程序的执行过程. 从这个框图很容易大体上了 ...
- Java虚拟机工作原理具体解释
一.类载入器 首先来看一下java程序的运行过程. 从这个框图非常easy大体上了解java程序工作原理.首先,你写好java代码,保存到硬盘其中.然后你在命令行中输入 javac YourClass ...
- springmvc工作原理和环境搭建
SpringMVC工作原理 上面的是springMVC的工作原理图: 1.客户端发出一个http请求给web服务器,web服务器对http请求进行解析,如果匹配DispatcherServle ...
随机推荐
- c# 服务
注:服务里的timer System.Timers.Timer time=new System.Timers.Timer(); time.Interval = 3000; //设置计时器事件间隔执 ...
- Spring核心——设计模式与IoC
“Spring”——每一个Javaer开发者都绕不开的字眼,从21世纪第一个十年国内异常活跃的SSH框架,到现在以Spring Boot作为入口粘合了各种应用.Spring现在已经完成了从web入口到 ...
- laravel compact的用法
向视图中传递变量 我们在开发web应用当中,通常都不是为了写静态页面而生的,我们需要跟数据打交道,那么这个时候,问题就来了,在一个MVC的框架中,怎么将数据传给视图呢?比如我们要在 ArticleCo ...
- [HTML/CSS]导航栏的下划线跟随效果
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...
- cf1132G. Greedy Subsequences(线段树)
题意 题目链接 Sol 昨天没想到真是有点可惜了. 我们考虑每个点作为最大值的贡献,首先预处理出每个位置\(i\)左边第一个比他大的数\(l\),显然\([l + 1, i]\)内的数的后继要么是\( ...
- 【代码笔记】Web-Javascript-Javascript typeof
一,效果图. 二,代码. <!DOCTYPE html> <html> <head> <meta charset="utf-8"> ...
- LockSupport的源码实现原理以及应用
一.为什么使用LockSupport类 如果只是LockSupport在使用起来比Object的wait/notify简单, 那还真没必要专门讲解下LockSupport.最主要的是灵活性. 上边的例 ...
- 性能测试 基于Python结合InfluxDB及Grafana图表实时采集Linux多主机或Docker容器性能数据
基于Python结合InfluxDB及Grafana图表实时采集Linux多主机性能数据 by:授客 QQ:1033553122 实现功能 1 测试环境 1 环境搭建 3 使用前提 3 使用方法 ...
- C# 利用ReportViewer生成报表
本文主要是利用微软自带的控件ReportViewer进行报表设计的小例子,仅供学习分享使用,如有不足之处,还请指正. 涉及知识点: ReportViewer :位于Microsoft.Reportin ...
- View体系第一篇:基础
View体系的学习内容为学习刘望舒先生博客总结的内容,大家可到他的博客看到更详细的内容. 一.view之间的继承关系 Viewground用来包裹其他view.在平常的使用中,我们不会直接用到View ...