为方便演示, 所有处理逻辑全部放在Controller完成, 不再写Service等各层接口及实现. 如需在Service层处理, 思路及方法也是完全一样的.
先说前台. 运行以后就是这样子的. 一个非常简单的表单页面, 两个文件上传按钮, 一个提交
其中单个文件上传, 即只能选择一个文件, 无法同时选择多个
相对的, 多个文件就是可以同时选择多个文件了
文件选择以后就是这个样子
代码如下: 一个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的值完全符合