让我们捋一捋上传和下载

开发 前端
当form表单的enctype属性为multipart/form-data时,浏览器就会采用二进制流来处理表单数据,服务器端就会对文件上传的请求进行解析处理。

[[433275]]

GitHub:https://github.com/nateshao/ssm/tree/master/111-springmvc-file-upload

1. 文件上传

文件上传概述

“多数文件上传都是通过表单形式提交给后台服务器的,因此,要实现文件上传功能,就需要提供一个文件上传的表单,而该表单必须满足以下3个条件:

  • form表单的method属性设置为post;
  • form表单的enctype属性设置为multipart/form-data;
  • 提供< input type="file" name="filename" />的文件上传输入框。

文件上传表单示例如下

  1. <form action="uploadUrl" method="post" enctype="multipart/form-data"
  2.  
  3.   <input type="file" name="filename" multiple="multiple" /> 
  4.  
  5.   <input type="submit" value="文件上传" /> 
  6.  
  7. </form> 

multiple属性是HTML5中新属性,可实现多文件上传

当form表单的enctype属性为multipart/form-data时,浏览器就会采用二进制流来处理表单数据,服务器端就会对文件上传的请求进行解析处理。Spring MVC通过MultipartResolver实现文件上传功能。MultipartResolver是一个接口对象,需要通过它的实现CommonsMultipartResolver来完成文件上传工作。

MultipartResolver配置示例如下:

  1. <bean id="multipartResolver"                 
  2.      class="org.springframework.web.multipart.commons.CommonsMultipartResolver"
  3.     设置请求编码格式,必须与JSP中的pageEncoding属性一致,默认为ISO-8859-1 
  4.     <property name="defaultEncoding" value="UTF-8" /> 
  5.     设置允许上传文件的最大值(2M),单位为字节 
  6.     <property name="maxUploadSize" value="2097152" /> 
  7.              ... 
  8. </bean> 

在前面的配置代码中,除配置了CommonsMultipartResolver类外,还通过< property>元素配置了编码格式以及允许上传文件的大小。通过< property>元素可以对文件解析器类CommonsMultipartResolver的如下属性进行配置:

  • maxUploadSize:上传文件最大长度(以字节为单位);
  • maxInMemorySize:缓存中的最大尺寸;
  • defaultEncoding:默认编码格式;
  • resolveLazily:推迟文件解析,以便在Controller中捕获文件大小异常。

注意:因为MultipartResolver接口的实现类CommonsMultipartResolver内部是引用multipartResolver字符串获取该实现类对象并完成文件解析的,所以在配置CommonsMultipartResolver时必须指定该Bean的id为multipartResolver。

由于CommonsMultipartResolver是Spring MVC内部通过Apache Commons FileUpload技术实现的,所以Spirng MVC的文件上传还需要依赖Apache Commons FileUpload的组件,即需要导入支持文件上传的相关JAR包。

  • commons-fileupload-1.3.2.jar
  • lcommons-io-2.5.jar

当完成页面表单和文件上传解析器的配置后,在Controller中编写文件上传的方法即可实现文件上传,其代码如下所示:

  1.  /** 
  2.      * 执行文件上传 
  3.      * @param name 
  4.      * @param uploadfile 
  5.      * @param request 
  6.      * @return 
  7.      */ 
  8. @RequestMapping("/fileUpload"
  9. public String handleFormUpload(@RequestParam("name") String name
  10.                                    @RequestParam("uploadfile") List<MultipartFile> uploadfile,//使用MultipartFile 绑定接收上传文件 
  11.                                    HttpServletRequest request) { 
  12.         // 判断所上传文件是否存在 
  13.         if (!uploadfile.isEmpty() && uploadfile.size() > 0) { 
  14.             //循环输出上传的文件 
  15.             for (MultipartFile file : uploadfile) { 
  16.                 // 获取上传文件的原始名称 
  17.                 String originalFilename = file.getOriginalFilename(); 
  18.                 // 设置上传文件的保存地址目录 
  19.                 String dirPath = 
  20.                         request.getServletContext().getRealPath("/upload/"); 
  21.                 File filePath = new File(dirPath); 
  22.                 // 如果保存文件的地址不存在,就先创建目录 
  23.                 if (!filePath.exists()) { 
  24.                     filePath.mkdirs(); 
  25.                 } 
  26.                 // 使用UUID重新命名上传的文件名称(上传人_uuid_原始文件名称) 
  27.                 String newFilename = name + "_" + UUID.randomUUID() + 
  28.                         "_" + originalFilename; 
  29.                 try { 
  30.                     // 使用MultipartFile接口的方法完成文件上传到指定位置 
  31.                     file.transferTo(new File(dirPath + newFilename)); 
  32.                 } catch (Exception e) { 
  33.                     e.printStackTrace(); 
  34.                     return "error"
  35.                 } 
  36.             } 
  37.             // 跳转到成功页面 
  38.             return "success"
  39.         } else { 
  40.             return "error"
  41.         } 
  42.     } 

在上述代码中,包含一个MultipartFile接口类型的参数file,上传到程序中的文件是被封装在该参数中的。

org.springframework.web.multipart.MultipartFile接口中提供了获取上传文件、文件名称等方法,如下表所示:

代码实现:

fileUpload.jsp

  1. <%@ page language="java" contentType="text/html; charset=UTF-8" 
  2.          pageEncoding="UTF-8" %> 
  3. <!DOCTYPE html> 
  4. <html> 
  5. <head> 
  6.     <meta http-equiv="Content-Type" content="text/html; charset=UTF-8"
  7.     <title>文件上传</title> 
  8.     <script> 
  9.         // 判断是否填写上传人并已选择上传文件 
  10.         function check() { 
  11.             var name = document.getElementById("name").value; 
  12.             var file = document.getElementById("file").value; 
  13.             if (name == "") { 
  14.                 alert("填写上传人"); 
  15.                 return false
  16.             } 
  17.             if (file.length == 0 || file == "") { 
  18.                 alert("请选择上传文件"); 
  19.                 return false
  20.             } 
  21.             return true
  22.         } 
  23.     </script> 
  24. </head> 
  25. <body> 
  26. <form action="${pageContext.request.contextPath }/fileUpload" 
  27.       method="post" enctype="multipart/form-data" onsubmit="return check()"
  28.     上传人:<input id="name" type="text" name="name"/><br/> 
  29.     请选择文件:<input id="file" type="file" name="uploadfile" 
  30.                  multiple="multiple"/><br/> 
  31.     <input type="submit" value="上传"/> 
  32. </form> 
  33. </body> 
  34. </html> 

2. 文件下载

文件下载就是将文件服务器中的文件下载到本机上。在Spring MVC环境中,实现文件下载大致可分为如下两个步骤:

在客户端页面使用一个文件下载的超链接,该链接的href属性要指定后台文件下载的方法以及文件名(需要先在文件下载目录中添加了一个名称为“1.jpg”的文件)。

  1. <a href="${pageContext.request.contextPath }/download?filename=1.jpg"
  2.      文件下载  
  3. </a> 

在后台使用Spring MVC提供的ResponseEntity类型对象完成文件下载,使用它可以很方便的定义返回的HttpHeaders对象和HttpStatus对象,通过对这两个对象的设置,即可完成下载文件时所需的配置信息。

  1. @RequestMapping("/download"
  2. public ResponseEntity<byte[]> fileDownload(HttpServletRequest request, 
  3.                                                String filename) throws Exception { 
  4.         // 指定要下载的文件所在路径 
  5.         String path = request.getServletContext().getRealPath("/upload/"); 
  6.         // 创建该文件对象 
  7.         File file = new File(path + File.separator + filename); 
  8.         // 对文件名编码,防止中文文件乱码 
  9.         filename = this.getFilename(request, filename); 
  10.         // 设置响应头 
  11.         HttpHeaders headers = new HttpHeaders(); 
  12.         // 通知浏览器以下载的方式打开文件 
  13.         headers.setContentDispositionFormData("attachment", filename); 
  14.         // 定义以流的形式下载返回文件数据 
  15.         headers.setContentType(MediaType.APPLICATION_OCTET_STREAM); 
  16.         // 使用Sring MVC框架的ResponseEntity对象封装返回下载数据 
  17.         return new ResponseEntity<byte[]>(FileUtils.readFileToByteArray(file), 
  18.                 headers, HttpStatus.OK); 

文件下载中的ResponseEntity对象有些类似前面章节中的@ResponseBody注解,它用于直接返回结果对象。

响应头信息中的MediaType代表的是Interner Media Type(即互联网媒体类型),也叫做MIME类型,MediaType.APPLICATION_OCTET_STREAM的值为application/octet-stream,即表示以二进制流的形式下载数据;

HttpStatus类型代表的是Http协议中的状态,示例中的HttpStatus.OK表示200,即服务器已成功处理了请求。

当对中文名文件下载时会怎样?

当对中文名称的文件进行下载时,因为各个浏览器内部转码机制的不同,就会出现不同的乱码以及解析异常问题。

如何解决中文名文件下载乱码问题呢?

为了解决浏览器中文件下载时中文名称的乱码问题,可以在前端页面发送请求前先对中文名进行统一编码,然后在后台控制器类中对文件名称进行相应的转码。

在下载页面中对中文文件名编码。可以使用Servlet API中URLEncoder.encoder(String s, String enc)方法将中文转为UTF-8编码。

  1. <%@ page language="java" contentType="text/html; charset=UTF-8" pageEncoding="UTF-8"%> 
  2.   <%@page import="java.net.URLEncoder"%> 
  3.   ... 
  4.   <body> 
  5.       <a href="${pageContext.request.contextPath }/download?filename=<%=URLEncoder.encode(“  
  6.         壁纸.jpg", "UTF-8")%>"
  7.  中文名称文件下载 </a> 
  8.     </body> 
  9. </html> 

在控制器类中编写对中文名文件下载时进行转码编码的方法。

  1. public String getFilename(HttpServletRequest request,String filename) throws Exception {  
  2.           String[] IEBrowserKeyWords = {"MSIE""Trident""Edge"};   
  3.           String userAgent = request.getHeader("User-Agent");   
  4.           for (String keyWord : IEBrowserKeyWords) {  
  5.                 if (userAgent.contains(keyWord)) {  
  6.                         return URLEncoder.encode(filename, "UTF-8"); 
  7.                 } 
  8.           }   
  9.           return new String(filename.getBytes("UTF-8"), "ISO-8859-1");   
  10. }  

总结

本章主要对Spring MVC环境下的文件上传和下载进行了详细讲解。

首先讲解了如何实现文件上传,并通过一个应用案例来演示文件上传功能的实现;

然后讲解了非中文名称文件下载的实现过程,以及中文名称文件下载的实现过程。

通过本章的学习,我们要学会如何在Spring MVC环境下进行文件上传和下载,并能够掌握中文名称文件下载时乱码的解决方案。

 

责任编辑:武晓燕 来源: 程序员千羽
相关推荐

2021-05-17 10:50:15

系统调用内核

2021-07-07 22:27:54

磁盘分区硬盘

2020-05-18 14:12:41

PostgreSQLDB架构数据库

2020-12-18 06:09:07

Java浅拷贝深拷贝

2023-03-15 10:38:55

2021-03-10 10:00:31

Go语言strconv包类型转换工具

2023-11-28 12:42:56

数据分析管理

2018-09-03 13:39:43

手机苹果iPhone

2021-07-27 05:32:22

CSS 技巧方位与顺序

2021-04-01 06:23:24

CSS33D3D Web 动画

2021-03-11 07:14:01

Epoll原理线程

2021-07-12 23:21:52

MyISAM引擎InnoDB

2022-08-17 11:36:18

Vue3插件

2021-03-04 08:06:13

Java代理机制

2021-04-23 10:38:52

Spring BootSpringMVC源码

2022-09-13 10:40:48

Flowable功能Spring

2021-08-02 12:50:45

sessiontokenJava

2021-02-03 12:47:09

Spring Boot应用监控

2021-11-26 07:45:37

编码GBK字符

2023-07-13 08:12:26

ControllerSpring管理
点赞
收藏

51CTO技术栈公众号