在了解Spring 文件上传执行流程之前,我们必须知道两点:

1.Spring 文件上传是基于common-fileUpload 组件的,所以,文件上传必须引入此包
2.Spring 文件上传需要在XML中配置文件上传解析器,不然程序无法执行判别

带领着这几个问题去分析Spring 文件上传:

1.文件上传解析器什么时候被初始化,在哪里被DispatcherServlet 引用
2.如何判别我们的请求上传文件的请求
3.传统的fileUpload与Spring被包装后的文件上传比较

问题抛出: 1.文件上传解析器什么时候被初始化调用的,在哪里被DispatcherServlet 引用:

  在我们在XML 中配置了如下:这个时候,MultipartResolver 最终会在Spring 实例化be an 的时候初始化,初始化的步骤会在第三个问题中一起看

<!-- 上传拦截,如最大上传值及最小上传值 -->
<bean id="multipartResolver" class="org.springframework.web.multipart.commons.CommonsMultipartResolver" >
<property name="maxUploadSize">
<value>524288000</value>
</property>
<property name="maxInMemorySize">
<value>4096</value>
</property>
<property name="defaultEncoding">
<value>utf-8</value>
</property>
</bean>

  在DispatcherServlet 中如何被引用的呢?

  聪明的你知道servlet 生命周期 init() 方法,只会被执行一次,所有的初始化动作在这里完成(org.springframework.web.servlet.DispatcherServlet),最终会调用如下的方法:  

  哈哈,这就是Spring 九大组件,没一个init..()方法就是一个组件,其中第一个就是初始化MultipartResolver

  所以我们XML 配置的id 可不是瞎取的  

  第一个问题结束;

  问题抛出:2.如何判别我们的请求是Multipart 文件上传请求:

  在doDispatch 方法中有检测我们请求的方法:如下:

   我们点进去:(1) 检测是否是Multipart ,不是则返回当前的http request

         (2) 解析resolveMultipart(request) 返回默认的 DefaultMultipartHttpServletRequest

    检测逻辑进入到了org.apache.commons.fileupload.servlet.ServletFileUpload 检测POST 方法,判断ContentType 是multipart;  

  到这里我们的HttpServletRequest 就被包装成了 DefaultMultipartHttpServletRequest ,里面有文件的信息还有原始HttpServletRequest的信息,至于怎么获取文件的,看抛出的

  第三个问题;

    问题抛出:3  传统的fileUpload与Spring被包装后的文件上传比较:

  我们看一下使用FileUpload 上传的逻辑代码:

/**
* 在XML 中没有配置CommonsMultipartResolver解析器情况下
* 使用fileUpload 文件上传
*/ DiskFileItemFactory fileItemFactory=new DiskFileItemFactory();
//创建FileUpload
ServletFileUpload fileUpload =new ServletFileUpload(fileItemFactory);
//获取所有的文件,每一个文件信息都在FileItem 中
List<FileItem> fileItems = fileUpload.parseRequest(req);
FileItem fileItem = fileItems.get(0);//demo 只有一个文件
String fieldName = fileItem.getFieldName();
System.out.println(fieldName);
InputStream inputStream = fileItem.getInputStream();
long size = fileItem.getSize();
//.....上传逻辑,到ftp 腾讯云 等

  这种方式,在每一次上传的时候,都必须写这么一段的获取文件的逻辑

  Spring 在此之上做了一层封装,使用 CommonsMultipartResolver 进行解析(底层还是fileUolad):

  对象创建:

  创建 DiskFileItemFactory 以及ServletFileUpload 两个对象,与我们上面代码逻辑一样

  这样解析器就初始化完成了。

  我们看解析代码:

  进行parserequest(request) 将List<FileItem> 封装到了sprintg 自己的 MultiartParsingResult 对象中;

  想像的到,就是将List<FileItem> 遍历,进行组合

  每一个文件都是一个 CommonsMultipartFile,也就是我们在controller 方法定义的MultipartFile  其实就是CommonMultipartFile

  将MultiartParsingResult 参数融合到一块,就形成了 DefaultMultipartHttpServletRequest 对象;在这个对象里就可以拿到你上传到文件;

  介绍DefaultMultipartHttpServletRequest几个方法吧:

  @RequestMapping("upload")
public void upload( MultipartFile file, HttpServletRequest req)throws Exception{ DefaultMultipartHttpServletRequest request=(DefaultMultipartHttpServletRequest)req;
//得到所有的文件名
Iterator<String> fileNames = request.getFileNames();
//得到所有的文件名与文件流的映射ma p
MultiValueMap<String, MultipartFile> multiFileMap = request.getMultiFileMap();
//通过名字得到MultipartFile 对象,其实就是CommonsMultipartFile
MultipartFile file2 = request.getFile("file");
//得到MultipartFile 对象就可以得到输入流
InputStream fileStream = file.getInputStream();
//文件名
String filename = file.getOriginalFilename();
//文件后缀
String suff =filename.substring(filename.lastIndexOf(".")+1); }

  参数解析的时候,也就是用的上面这些方法,拿到MultipartFile 的;Spring参数解析请看我以后的文章

  嗯,就结束了,哈哈

Spring 文件上传MultipartFile 执行流程分析的更多相关文章

  1. Hadoop之HDFS原理及文件上传下载源码分析(下)

    上篇Hadoop之HDFS原理及文件上传下载源码分析(上)楼主主要介绍了hdfs原理及FileSystem的初始化源码解析, Client如何与NameNode建立RPC通信.本篇将继续介绍hdfs文 ...

  2. spring文件上传

    Spring文件上传 1,导包: <dependency> <groupId>javax.servlet</groupId> <artifactId>s ...

  3. Spring文件上传出错:java.lang.ClassCastException: org.apache.catalina.connector.Request

    java.lang.ClassCastException: org.apache.catalina.connector.RequestFacade cannot be cast to org.spri ...

  4. SpringMVC 文件上传 MultipartFile

    本的SpringMVC的搭建在我的上一篇文章里已经写过了,这篇文章主要说明一下如何使用SpringMVC进行表单上的文件上传以及多个文件同时上传的步骤 SpringMVC 基础教程 框架分析:http ...

  5. Spring文件上传配置

    增加依赖jar包 <dependency> <groupId>commons-fileupload</groupId> <artifactId>comm ...

  6. Spring 文件上传功能

    本篇文章,我们要来做一个Spring的文件上传功能: 1. 创建一个Maven的web工程,然后配置pom.xml文件,增加依赖: <dependency> <groupId> ...

  7. Hadoop之HDFS原理及文件上传下载源码分析(上)

    HDFS原理 首先说明下,hadoop的各种搭建方式不再介绍,相信各位玩hadoop的同学随便都能搭出来. 楼主的环境: 操作系统:Ubuntu 15.10 hadoop版本:2.7.3 HA:否(随 ...

  8. (转)Spring文件上传,包括一次选中多个文件

    背景: http://www.cnblogs.com/lixuwu/p/8495275.html已经实现了单文件的上传和下载,多文件的上传是另一种情景,这里记录下来 实现过程 先说前台. 运行以后就是 ...

  9. Spring文件上传Demo

    package com.smbea.controller; import java.io.File; import java.io.FileOutputStream; import java.io.I ...

随机推荐

  1. boolean表达式与在if条件中的运用

    1.boolean语句 boolean 类型的数据 有两个值 false 和true; 2.在if判断的条件语句中 例如: boolean arn =false  在if 判断的括号中可以表示为 !a ...

  2. [STM31F103]独立看门狗

    独立看门狗步骤: l 取消寄存器写保护: n IWDG_WriteAccessCmd(); l 设置独立看门狗的预分频系数,确定时钟: n IWDG_SetPrescaler(); l 设置看门狗重装 ...

  3. 架构师技能图谱 V1.2

    系统架构能力 基本理论 扩展性设计 可用性设计 可靠性设计 一致性设计 负载均衡设计 过载保护设计 灾难恢复和备份 协议设计 二进制协议 文本协议 接入层架构设计 DNS 轮询 动静态分离 静态化 反 ...

  4. LINUX 安装扩展(笔记)

    1.下载扩展包: 网址:http://pecl.php.net/ 2.解压安装包. 3.进入解压好的安装包. 4.使用php中的phpize (扩展模块) 使其生成configure. 5../con ...

  5. Spring获取URL相关信息

    获取请求的URL:request.getRequestURL().toString(); 获取上下文名称(项目名称):request.getContextPath()

  6. Spring Security开发Restful服务

    2-1开发环境安装 1.jdk8安装 2.安装STS    Spring Tool Suite实际上就是一个eclipse,只不过在此基础上又安装了一些插件 3.安装mysql 2-2代码结构介绍 打 ...

  7. 使用nexus3.x搭建maven私服

    前言 好久之前就想搭建maven仓库了,一直拖到了现在,也就是懒得动,现在终于是要付诸行动了.网上查了不少资料,好多博客都是关于2.x的搭建,我下载的是最新版的nexus,好多教程已经不能使用,以此记 ...

  8. kubelet工作原理

    在调度这一步完成后,Kubernetes 就需要负责将这个调度成功的 Pod,在宿主机上创建出来,并把它所定义的各个容器启动起来.这些,都是 kubelet 这个核心组件的主要功能. kubelet ...

  9. java Page分页显示

    //entity层实体类 import java.util.List; //分页展示 //相关属性:当前页,页大小(每页显示的条数),总页数,总条数,数据 //select * from t_user ...

  10. 腾讯、爱奇艺、优酷等vip视频在线解析

    http://vip.mist.xin 菠萝蜜TV全网VIP视频在线看 菠萝蜜tv http://jx.mist.xin 全网VIP视频在线解析接口 免费全网影视VIP视频vip会员免广告看电影!亲们 ...