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();}// 访问的urlString 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();}// 访问的urlString 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标签的nameMultipartFile 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();}// 访问的urlString 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 对象中。

原文地址:https://www.cnblogs.com/tengyunhao/p/7670293.html

SpringMVC工作原理之四:MultipartResolver相关推荐

  1. [Java] SpringMVC工作原理之四:MultipartResolver

    MultipartResolver 用于处理文件上传,当收到请求时 DispatcherServlet 的 checkMultipart() 方法会调用 MultipartResolver 的 isM ...

  2. SpringMVC工作原理详解

    点击上方"方志朋",选择"置顶或者星标" 你的关注意义重大! 先来看一下什么是 MVC 模式 MVC 是一种设计模式. MVC 的原理图如下: SpringMV ...

  3. SpringMVC工作原理之一:DispatcherServlet

    一.DispatcherServlet 处理流程 在整个 Spring MVC 框架中,DispatcherServlet 处于核心位置,它负责协调和组织不同组件完成请求处理并返回响应工作.在看 Di ...

  4. SpringMVC工作原理 1

    大家好,我是IT修真院深圳分院第十一期学员,一枚正直纯洁善良的JAVA程序员. 今天给大家分享一下,修真院官网JAVA任务二的一个知识点:SpringMVC工作原理 1.背景介绍 一:背景介绍 Jav ...

  5. SpringMVC→简介、MVC、SpringMVC工作原理、Maven搭建第一个SpringMVC、请求参数接收、重定向、文件上传、AJAX异步访问、请求参数接收绑定JSON、@注解及传参

    MVC SpringMVC工作原理 Maven搭建第一个SpringMVC 目录结构 web.xml *-servlet.xml Controller请求处理类 跳转页面 Maven运行服务器项目 浏 ...

  6. SpringMVC工作原理及源码解析

    SpringMVC工作原理及源码解析 一:SpringMVC原理图 二:SpringMVC的主要组件 1.前端控制器DispatcherServlet: 2.处理器映射器HandlerMapping: ...

  7. SpringMVC工作原理 侵立删

    转自:http://www.cnblogs.com/xiaoxi/ SpringMVC的工作原理图: SpringMVC原理 1.用户发送请求至前端控制器DispatcherServlet. 2. D ...

  8. SpringMVC工作原理

    一:SpringMVC的工作原理图 二:SpringMVC流程 用户发送请求至前端控制器DispatcherServlet. DispatcherServlet收到请求调用HandlerMapping ...

  9. dispatcherservlet发送2次请求_[Java] SpringMVC工作原理之一:DispatcherServlet

    一.DispatcherServlet 处理流程 在整个 Spring MVC 框架中,DispatcherServlet 处于核心位置,它负责协调和组织不同组件完成请求处理并返回响应工作.在看 Di ...

  10. SpringMVC工作原理的介绍

    1.原理图: 2.流程文字介绍: 1.用户向服务器发送请求,请求被Spring前端控制Servlet DispatcherServlet捕获: 2.DispatcherServlet对请求UR进行解析 ...

最新文章

  1. 广东时代互联---网络管理面试
  2. Windows10下通过anaconda安装tensorflow
  3. highcharts图标插件详解一
  4. Excel宏的自动运行设置
  5. [转载] python随笔2(列表的增删改查)
  6. windows系统查看80端口被占用的程序并结束该程序运行
  7. McAfee VirusScan Enterprise
  8. win8打印机显示服务器脱机,Win8连接局域网打印机失败解决方法汇总
  9. 数据库学习7 — 嵌套查询
  10. SoftWare Engineering -- WEEK.3
  11. icp相关数学,向量模的平方转换成矩阵相乘
  12. opencv-viz模块简单示例
  13. el-element布局控件layout中的el-row和el-col
  14. MODBUS-RTU数据帧格式、报文解析
  15. 【EI会议|检索稳定】2021信息、控制及自动化国际学术会议(ICICA 2021)
  16. ipxe无盘服务器,ipxe uefi pxe HTTP启动文件及启动菜单的个人体会
  17. 如何从win10中获取3D模型(GLB格式)
  18. 前端开源项目周报1226
  19. word超链接显示HYPERLINK
  20. 笔记本USB的PowerShare功能

热门文章

  1. Matlab识别拨号音,电话拨号音识别全解.ppt
  2. python输入一个包含若干自然数的列表_Python练习题
  3. LayerPagerDemo - 双层可拖拽式布局界面
  4. php收付同分账,php微信分账功能 —— app支付
  5. mysql 常见问题处理_mysql常见问题处理
  6. Java之美[从菜鸟到高手演变]之Java学习方法
  7. 详解安卓辅助功能服务AccessibilityService(无障碍服务,微信抢红包助手原理)
  8. 单臂路由之一,单网口软路由实现主路由功能,光猫或交换机剩余网口实现上网功能
  9. 【渝粤教育】电大中专电商运营实操 (4)作业 题库
  10. 查看SqlServer数据库上面脚本修改信息以及修改日志