Spring MVC 实现文件的上传和下载 (八) 目录

 完整的项目案例: springmvc.zip

Spring MVC 实现文件的上传和下载 (八)
目录

SpringMVC 中,文件的上传,是通过 MultipartResolver 实现的。 所以,如果要实现文件的上传,只要在 spring-mvc.xml 中注册相应的 MultipartResolver 即可。

MultipartResolver 的实现类有两个:

  1. CommonsMultipartResolver
  2. StandardServletMultipartResolver

两个的区别:

  1. 第一个需要使用 Apache 的 commons-fileupload 等 jar 包支持,但它能在比较旧的 servlet 版本中使用。
  2. 第二个不需要第三方 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>

Spring MVC 实现文件的上传和下载 (八)
目录

上传成功后转发的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>

Spring MVC 实现文件的上传和下载 (八)
目录

项目结构:

Spring MVC 实现文件的上传和下载 (八)
目录