完整的项目案例: springmvc.zip
目录
SpringMVC 中,文件的上传,是通过 MultipartResolver 实现的。 所以,如果要实现文件的上传,只要在 spring-mvc.xml 中注册相应的 MultipartResolver 即可。
MultipartResolver 的实现类有两个:
- CommonsMultipartResolver
- StandardServletMultipartResolver
两个的区别:
- 第一个需要使用 Apache 的 commons-fileupload 等 jar 包支持,但它能在比较旧的 servlet 版本中使用。
- 第二个不需要第三方 jar 包支持,它使用 servlet 内置的上传功能,但是只能在 Servlet 3 以上的版本使用。
第一个使用步骤:
/*CommonsMultipartResolver 上传用到的两个包*/ "commons-fileupload:commons-fileupload:1.3.1", "commons-io:commons-io:2.4"
如果是maven项目的话直接导入:
<dependency>
<groupId>commons-fileupload</groupId>
<artifactId>commons-fileupload</artifactId>
<version>1.3.1</version>
</dependency> dispatcher-servlet.xml配置:
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:context="http://www.springframework.org/schema/context"
xmlns:mvc="http://www.springframework.org/schema/mvc"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd http://www.springframework.org/schema/mvc http://www.springframework.org/schema/mvc/spring-mvc.xsd"> <context:component-scan base-package="edu.nf.ch08.controller"/> <mvc:annotation-driven/> <mvc:default-servlet-handler/>
<!-- 文件上传有两种方式,一种基于Servlet3.0的上传,一种基于
commons-upload上传,如果使用Servlet3.0的上传方式,可以
不需要配置MultipartResolver,Spring默认会注册一个
StandardServletMultipartResolver。只需要在web.xml中
启用<multipart-config>。
如果想使用commons-upload,那么需要配置一个CommonsMultipartResolver,
且指定bean的id为multipartResolver-->
<!-- 这里使用commons-upload-->
<bean id="multipartResolver" class="org.springframework.web.multipart.commons.CommonsMultipartResolver">
<!-- 限制文件上传的总大小(单位:字节),不配置此属性默认不限制 -->
<property name="maxUploadSize" value="104857600"/>
<!-- 设置文件上传的默认编码-->
<property name="defaultEncoding" value="utf-8"/>
</bean> <bean class="org.springframework.web.servlet.view.InternalResourceViewResolver">
<property name="prefix" value="/WEB-INF/jsp/"/>
<property name="suffix" value=".jsp"/>
</bean>
</beans>
web.xml:
<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns="http://xmlns.jcp.org/xml/ns/javaee"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/javaee http://xmlns.jcp.org/xml/ns/javaee/web-app_4_0.xsd"
version="4.0"> <!-- 请求总控器 -->
<servlet>
<servlet-name>dispatcher</servlet-name>
<servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
<init-param>
<param-name>contextConfigLocation</param-name>
<param-value>classpath:dispatcher-servlet.xml</param-value>
</init-param>
</servlet>
<servlet-mapping>
<servlet-name>dispatcher</servlet-name>
<url-pattern>/</url-pattern>
</servlet-mapping> </web-app>
后台java(上传、下载)处理代码:
package edu.nf.ch08.controller; import org.apache.commons.io.FileUtils;
import org.springframework.core.io.InputStreamResource;
import org.springframework.http.HttpHeaders;
import org.springframework.http.HttpStatus;
import org.springframework.http.MediaType;
import org.springframework.http.ResponseEntity;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.multipart.MultipartFile;
import org.springframework.web.servlet.ModelAndView; import java.io.File;
import java.io.IOException;
import java.io.InputStream;
import java.net.URLEncoder; /**
* @author wangl
* @date 2018/11/2
*/
@Controller
public class UploadController { /**
* 文件上传只需要Spring传入一个MultipartFile对象即可,
* 这个对象可以获取文件相关上传的信息。
* 一个MultipartFile表示单个文件上传,当需要上传多个文件时
* 只需要声明为MultipartFile[]数组即可。
* @return
*/
@PostMapping("/upload")
public ModelAndView upload(MultipartFile file){
//获取当前系统用户目录
String home = System.getProperty("user.home");
//指定上传的文件夹目录
File uploadDir = new File(home + "/files");
//如果目录不存在,则创建
if(!uploadDir.exists()){
uploadDir.mkdir();
}
//获取上传的文件名
String fileName = file.getOriginalFilename();
//构建一个完整的文件上传对象
File uploadFile = new File(uploadDir.getAbsolutePath() + "/" + fileName);
try {
//通过transferTo方法进行上传
file.transferTo(uploadFile);
} catch (IOException e) {
e.printStackTrace();
throw new RuntimeException(e.getMessage());
}
//将文件名存入Model,转发到index页面
ModelAndView mv = new ModelAndView("index");
mv.addObject("fileName", fileName);
return mv;
} /**
* 文件下载1
* 读取服务器本地文件并封装为ResponseEntity对象
* 响应客户端,ResponseEntity封装一个字节数组。
*
* 注意:如果文件很大,那么读入内存的字节数组就会很大,这时很容易引起内存溢出。
* 因此,这种方法不太适合下载大文件使用
* @param fileName 文件名
* @return
*/
@GetMapping("/download")
public ResponseEntity<byte[]> download(String fileName){
//依据文件名构建本地文件路径
String filePath = System.getProperty("user.home") + "/files/" + fileName;
//依据文件路径构建File对象
File file = new File(filePath);
//创建响应头对象,设置响应信息
HttpHeaders headers = new HttpHeaders();
try {
//对文件名进行重新编码,防止在响应头中出现中文乱码
String headerFileName = URLEncoder.encode(fileName,"UTF-8");
//设置响应内容处理方式为附件,并指定文件名
headers.setContentDispositionFormData("attachment", headerFileName);
//设置响应头类型为application/octet-stream,表示是一个流类型
headers.setContentType(MediaType.APPLICATION_OCTET_STREAM);
//将文件转换成字节数组
byte[] bytes = FileUtils.readFileToByteArray(file);
//创建ResponseEntity对象(封装文件字节数组、响应头、响应状态码)
ResponseEntity<byte[]> entity = new ResponseEntity<>(bytes, headers, HttpStatus.CREATED);
//最后将整个ResponseEntity对象返回给DispatcherServlet
return entity;
} catch (Exception e) {
e.printStackTrace();
throw new RuntimeException("文件下载失败");
}
} /**
* 文件下载2(主要解决下载大文件)
* 读取服务器本地文件并封装为ResponseEntity对象
* 响应客户端,ResponseEntity封装一个InputStreamResource
* @param fileName 文件名
* @return
*/
@GetMapping("/download2")
public ResponseEntity<InputStreamResource> download2(String fileName){
//依据文件名构建本地文件路径
String filePath = System.getProperty("user.home") + "/files/" + fileName;
//依据文件路径构建File对象
File file = new File(filePath);
//创建响应头对象,设置响应信息
HttpHeaders headers = new HttpHeaders();
try {
//对文件名进行重新编码,防止在响应头中出现中文乱码
String headerFileName = URLEncoder.encode(fileName,"UTF-8");
//设置响应内容处理方式为附件,并指定文件名
headers.setContentDispositionFormData("attachment", headerFileName);
//设置响应头类型为application/octet-stream,表示是一个流类型
headers.setContentType(MediaType.APPLICATION_OCTET_STREAM);
//打开一个输入流
InputStream inputStream = FileUtils.openInputStream(file);
//创建InputStreamResource封装输入流对象,用于读取服务器文件
InputStreamResource resource = new InputStreamResource(inputStream);
//创建ResponseEntity对象(InputStreamResource、响应头、响应状态码)
ResponseEntity<InputStreamResource> entity = new ResponseEntity<>(resource, headers, HttpStatus.CREATED);
//最后将整个ResponseEntity对象返回给DispatcherServlet
return entity;
} catch (Exception e) {
e.printStackTrace();
throw new RuntimeException("文件下载失败");
}
}
}
上传文件的网页html:
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>Title</title>
</head>
<body>
<h1>文件上传</h1>
<!-- 当有文件上传时,表单的enctype必须设置为multipart/form-data -->
<form method="post" action="upload" enctype="multipart/form-data">
File:<input type="file" name="file"/><br/>
<input type="submit" value="submit"/>
</form>
</body>
</html>
上传成功后转发的jsp(下载文件)页面:
<%--
Created by IntelliJ IDEA.
User: wangl
Date: 2018/11/2
Time: 09:56
To change this template use File | Settings | File Templates.
--%>
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
<title>Title</title>
</head>
<body>
<a href="download2?fileName=${fileName}">${fileName}</a>
</body>
</html>
项目结构: