为方便演示, 所有处理逻辑全部放在Controller完成, 不再写Service等各层接口及实现. 如需在Service层处理, 思路及方法也是完全一样的.


先说前台. 运行以后就是这样子的. 一个非常简单的表单页面, 两个文件上传按钮, 一个提交

spring通过文件流方式上传文件 spring多文件上传_上传


其中单个文件上传, 即只能选择一个文件, 无法同时选择多个

spring通过文件流方式上传文件 spring多文件上传_spring通过文件流方式上传文件_02



相对的, 多个文件就是可以同时选择多个文件了

spring通过文件流方式上传文件 spring多文件上传_spring通过文件流方式上传文件_03


文件选择以后就是这个样子

spring通过文件流方式上传文件 spring多文件上传_spring通过文件流方式上传文件_04


代码如下: 一个form, 文件上传就是一个<input>输入, 属性type="file". 此时只能选择单个文件. 而后面加一个multiple, 即可同时选择多个文件


action属性中的路径后缀为.htm, 是因为我的环境中配置了映射, 所以要在Controller中指定的路径后添加一个.htm后缀, 否则系统会报404. 如果没有配置该项则无需添加后缀


<body>
	<form action="${pageContext.request.contextPath}/test/upload.htm" enctype="multipart/form-data" method="post">
		单个文件: <input type="file" name="fileTest"><br/>
		多个文件: <input type="file" name="fileList" multiple/></br/>
		<input type="submit" value="提交" />
	</form>
</body>



另一点需要注意的是, 要实现文件上传, form中必须指定属性enctype="multipart/form-data". method属性为"post"


前台就这些东西了, 没什么特殊的. 然后再看后台


首先Spring配置文件中加这么一个bean


<bean id="multipartResolver"
	class="org.springframework.web.multipart.commons.CommonsMultipartResolver">
	<!-- 默认编码 -->
	<property name="defaultEncoding" value="utf-8" />
	<!-- 文件大小最大值 -->
	<property name="maxUploadSize" value="10485760" />
	<!-- 内存中的最大值 -->
	<property name="maxInMemorySize" value="40960" />
</bean>


也可以在代码中直接创建对象


CommonsMultipartResolver commonsMultipartResolver = new   CommonsMultipartResolver(request.getSession().getServletContext());

但是个人认为配置以后使用比较方便, 各位根据实际需要来吧


其中maxUploadSize属性用来设计上传文件的最大值, 单位是字节. 注意这里是总的上传限制, 比如设置为10M, 上传了4个3M的文件. 虽然单个文件都在10M以内, 但是总大小已经超过10M限制, 会抛出MaxUploadSizeExceededException异常



package com.test.controller;

import java.io.File;
import java.io.IOException;
import java.util.Iterator;
import java.util.List;

import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.ResponseBody;
import org.springframework.web.multipart.MultipartFile;
import org.springframework.web.multipart.MultipartHttpServletRequest;

/**
 * 文件上传测试类
 */
@Controller
@RequestMapping("/test")
public class FileUploadController {

    @ResponseBody
    @RequestMapping(value="upload")
    public void testUpload(MultipartHttpServletRequest request) throws IOException {
        /*
         * MultipartHttpServletRequest: 继承于HttpServletRequest以及MultipartRequest.
         * 其中MultipartRequest中定义了相关的访问操作. MultipartHttpServletRequest重写
         * 了HttpServletRequest中的方法, 并进行了扩展. 如果以HttpServletRequest来接收参
         * 数, 则需要先将其转为MultipartHttpServletRequest类型
         * MultipartHttpServletRequest request = (MultipartHttpServletRequest) HttpServletRequest;
         */
        
        /*
         * 再说回刚才的form, 假设我们在单个文件选框中上传了文件1, 多个文件选框中上传了文件2, 3, 4.
         * 那么对于后台接收到的, 可以这么理解, 就是一个Map的形式(实际上它后台真的是以Map来存储的).
         * 这个Map的Key是什么呢? 就是上面<input>标签中的name=""属性. Value则是我们刚才上传的
         * 文件, 通过下面的示例可以看出每一个Value就是一个包含对应文件集合的List
         * 
         * 传到后台接收到的Map就是这样:
         * fileTest: 文件1
         * fileList: 文件2, 文件3, 文件4
         * 
         * 虽然从方法名的表面意义来看是得到文件名, 但实际上这个文件名跟上传的文件本身并没有什么关系.
         * 刚才说了这个Map的Key就是<input>标签中的name=""属性, 所以得到的也就是这个属性的值
         */
        Iterator<String> fileNames = request.getFileNames();
        
        while (fileNames.hasNext()) {
            
            //把fileNames集合中的值打出来
            String fileName=fileNames.next();
            System.out.println("fileName: "+fileName);
            
            /*
             * request.getFiles(fileName)方法即通过fileName这个Key, 得到对应的文件
             * 集合列表. 只是在这个Map中, 文件被包装成MultipartFile类型
             */
            List<MultipartFile> fileList=request.getFiles(fileName);
            
            if (fileList.size()>0) {
                
                //遍历文件列表
                Iterator<MultipartFile> fileIte=fileList.iterator();
                
                while (fileIte.hasNext()) {
                    
                    //获得每一个文件
                    MultipartFile multipartFile=fileIte.next();
                    
                    //获得原文件名
                    String originalFilename = multipartFile.getOriginalFilename();
                    System.out.println("originalFilename: "+originalFilename);
                    
                    //设置保存路径. 
                    String path ="G:/testUpload/";
                    
                    //检查该路径对应的目录是否存在. 如果不存在则创建目录
                    File dir=new File(path);
                    if (!dir.exists()) {
                        dir.mkdirs();
                    }
                    
                    String filePath = path + originalFilename;
                    System.out.println("filePath: "+filePath);
                    
                    //保存文件
                    File dest = new File(filePath);
                    if (!(dest.exists())) {
                        /*
                         * MultipartFile提供了void transferTo(File dest)方法,
                         * 将获取到的文件以File形式传输至指定路径.
                         */
                        multipartFile.transferTo(dest);
                        
                        /*
                         * 如果需对文件进行其他操作, MultipartFile也提供了
                         * InputStream getInputStream()方法获取文件的输入流
                         * 
                         * 例如下面的语句即为通过
                         * org.apache.commons.io.FileUtils提供的
                         * void copyInputStreamToFile(InputStream source, File destination)
                         * 方法, 获取输入流后将其保存至指定路径
                         */
                        //FileUtils.copyInputStreamToFile(multipartFile.getInputStream(), dest);
                    }
                    
                    //MultipartFile也提供了其他一些方法, 用来获取文件的部分属性
                    
                    //获取文件contentType
                    String contentType=multipartFile.getContentType();
                    System.out.println("contentType: "+contentType);
                    
                    /*
                     * 获取name
                     * 其实这个name跟上面提到的getFileName值是一样的,
                     * 就是Map中Key的值. 即前台页面<input>中name=""
                     * 属性. 但是上面的getFileName只是得到这个Map的Key,
                     * 而Spring在处理上传文件的时候会把这个值以name属性
                     * 记录到对应的每一个文件. 如果需要从文件层面获取这个
                     * 值, 则可以使用该方法 
                     */
                    String name=multipartFile.getName();
                    System.out.println("name: "+name);
                    
                    //获取文件大小, 单位为字节
                    long size=multipartFile.getSize();
                    System.out.println("size: "+size);
                    
                    System.out.println("---------------------------------------------------");
                }
            }
        }
    }
}



点击上传后控制台打印出的属性:

fileName: fileTest
 originalFilename: jquery-validation-1.14.0.zip
 filePath: G:/testUpload/jquery-validation-1.14.0.zip
 contentType: application/octet-stream
 name: fileTest
 size: 879139
 ---------------------------------------------------
 fileName: fileList
 originalFilename: ueditor_release_ueditor1_4_3_1-src.zip
 filePath: G:/testUpload/ueditor_release_ueditor1_4_3_1-src.zip
 contentType: application/octet-stream
 name: fileList
 size: 4103285
 ---------------------------------------------------
 originalFilename: zookeeper-3.4.8.tar.gz
 filePath: G:/testUpload/zookeeper-3.4.8.tar.gz
 contentType: application/gzip
 name: fileList
 size: 22261552
 ---------------------------------------------------
 originalFilename: zTree_v3-master.zip
 filePath: G:/testUpload/zTree_v3-master.zip
 contentType: application/octet-stream
 name: fileList
 size: 829706
 ---------------------------------------------------

查看刚才上传的文件属性, 与打印出的size的值完全符合

spring通过文件流方式上传文件 spring多文件上传_spring通过文件流方式上传文件_05