文件的上传和下载,是非常常见的功能。很多的系统中,或者软件中都经常使用文件的上传和下载。

比如:QQ头像,就使用了上传。

邮箱中也有附件的上传和下载功能。

OA系统中审批有附件材料的上传。

1、文件的上传介绍(重点)

  1. 要有一个 form 标签,method=post 请求
  2. form 标签的 encType 属性值必须为 multipart/form-data 值
  3. 在 form 标签中使用 input type=file 添加上传的文件
  4. 编写服务器代码(Servlet 程序)接收,处理上传的数据。

encType=multipart/form-data 表示提交的数据,以多段(每一个表单项一个数据段)的形式进行拼接,然后以二进制流的形式发送给服务器

1.1、文件上传,HTTP协议的说明

9、文件的上传与下载 - 图1

注意:谷歌浏览器中上传的文件的数据显示的是空行,但服务器可以接收到数据

1.2、commons-fileupload.jar 常用 API 介绍说明

commons-fileupload.jar 需要依赖 commons-io.jar 这个包,所以两个包我们都要引入。

第一步,就是需要导入两个 jar 包: (fileupload包依赖io包)

  • commons-fileupload-1.2.1.jar
  • commons-io-1.4.jar

两个jar包中常用的类 (导入的jar包是commons的)

ServletFileUpload类,用于解析上传的数据

    1. public static final boolean isMultipartContent(HttpServletRequest request)
    • 如果上传的数据是多段的形式,返回true,只有多段的数据才是文件上传的
    1. public ServletFileUpload()
    • 空参构造器
    1. public ServletFileUpload(FileItemFactory fileItemFactory)
    • 参数为工厂实现类的构造器
    1. public List parseRequest(HttpServletRequest request)
    • 解析上传的数据,返回包含每一个表单项的List集合

FileItem类,表示每一个表单项

    1. public boolean isFormField()
    • 如果当前表单项是普通表单项,返回true,如果上传的文件类型则返回false
    1. public String getFieldName()
    • 获取当前表单项的name属性值
    1. public String getString()
    • 获取当前表单项的value属性值,参数为”UTF-8”可解决乱码问题
    1. public String getName()
    • 获取上传的文件名
    1. public void write(File file)
    • 将上传的文件写到参数File所指向的硬盘位置

代码实现:

upload.jsp

  1. <%--
  2. Created by IntelliJ IDEA.
  3. User: DELL
  4. Date: 2021/11/13
  5. Time: 22:13
  6. To change this template use File | Settings | File Templates.
  7. --%>
  8. <%@ page contentType="text/html;charset=UTF-8" language="java" %>
  9. <html>
  10. <head>
  11. <title>Title</title>
  12. </head>
  13. <body>
  14. <form action="http://localhost:8080/jsp/uploadServlet" method="post" enctype="multipart/form-data">
  15. 用户名:<input type="text" name="username" ><br>
  16. 头像:<input type="file" name="photo" ><br>
  17. <input type="submit" value="上传">
  18. </form>
  19. </body>
  20. </html>

UploadServlet.java

  1. package com.chang.servlet;
  2. import org.apache.commons.fileupload.FileItem;
  3. import org.apache.commons.fileupload.FileItemFactory;
  4. import org.apache.commons.fileupload.FileUploadException;
  5. import org.apache.commons.fileupload.disk.DiskFileItemFactory;
  6. import org.apache.commons.fileupload.servlet.ServletFileUpload;
  7. import javax.servlet.ServletException;
  8. import javax.servlet.http.HttpServlet;
  9. import javax.servlet.http.HttpServletRequest;
  10. import javax.servlet.http.HttpServletResponse;
  11. import java.io.File;
  12. import java.io.IOException;
  13. import java.util.List;
  14. public class UploadServlet extends HttpServlet {
  15. @Override
  16. protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
  17. // 先判断上传的数据是否多段数据(只有是多段的数据,才是文件上传的)
  18. if(ServletFileUpload.isMultipartContent(req)){
  19. // 创建FileItemFactory工厂实现类
  20. FileItemFactory fileItemFactory = new DiskFileItemFactory();
  21. // 创建用于解析上传数据的工具类ServletFileUpload类
  22. ServletFileUpload servletFileUpload = new ServletFileUpload(fileItemFactory);
  23. try {
  24. // 解析上传的数据,得到每一个表单项FileItem
  25. List<FileItem> list = servletFileUpload.parseRequest(req);
  26. // 循环判断,每一个表单项,是普通类型,还是上传的文件
  27. for (FileItem fileItem : list) {
  28. if (fileItem.isFormField()) {
  29. // 普通表单项
  30. System.out.println("表单项的name属性值:" + fileItem.getFieldName());
  31. // 参数UTF-8.解决乱码问题
  32. System.out.println("表单项的value属性值:" + fileItem.getString("UTF-8"));
  33. } else {
  34. // 上传的文件
  35. System.out.println("表单项的name属性值:" + fileItem.getFieldName());
  36. System.out.println("上传的文件名:" + fileItem.getName());
  37. fileItem.write(new File("e:\\" + fileItem.getName()));
  38. }
  39. }
  40. } catch (Exception e) {
  41. e.printStackTrace();
  42. }
  43. }
  44. }
  45. }

web.xml

  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <web-app xmlns="http://xmlns.jcp.org/xml/ns/javaee"
  3. xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  4. xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/javaee http://xmlns.jcp.org/xml/ns/javaee/web-app_4_0.xsd"
  5. version="4.0">
  6. <servlet>
  7. <servlet-name>UploadServlet</servlet-name>
  8. <servlet-class>com.chang.servlet.UploadServlet</servlet-class>
  9. </servlet>
  10. <servlet-mapping>
  11. <servlet-name>UploadServlet</servlet-name>
  12. <url-pattern>/uploadServlet</url-pattern>
  13. </servlet-mapping>
  14. </web-app>

2、文件的下载过程

9、文件的上传与下载 - 图2

  • 获取要下载的文件名
  • 获取要下载的文件类型
  • 将获取的文件类型告诉客户端
  • 告诉客户端收到的数据用于下载使用
  • 获取要下载的文件并回传给客户端

2.1、文件下载过程详解

  1. 获取要下载的文件名:使用String定义要下载的文件名
  2. 获取要下载的文件类型:

    1. 通过ServletContext`**getMimeType()**`参数是要下载的文件所在路径,返回值是String类型
  3. 将获取的文件类型告诉客户端:

    1. 通过response`**setContentType()**`参数是第二步的结果,无返回值
  4. 告诉客户端收到的数据用于下载使用(没有此步则内容直接显示在页面上):

    1. 通过response.`**setHeader()**`
    2. 参数是**"Content-Disposition", attachment; fileName=xxx.xxx"**
    3. 注意:
    4. Content-Disposition 响应头表示客户端收到的数据如何处理
    5. attachment 表示附件,用于下载
    6. filename 表示下载的文件名,可以与原文件名不同
  5. 获取要下载的文件并回传给客户端:

    1. 回传给客户端通过导入的io包的IOUtils.`copy(InputStream input, OutputStream output)`
    2. 通过ServletContext`getResourceAsStream()`参数是要下载的文件路径,得到输入流
    3. 通过response.`getOutputStream()`得到响应的输出流

2.2、中文名下载文件的乱码问题

原因

response.setHeader(“Content-Disposition”, “attachment; fileName=中文名.jpg”);

如果下载的文件是中文名,会发现下载的文件无法正常显示汉字,原因是响应头中不能有汉字

解决

  1. 当浏览器是IE浏览器或谷歌浏览器:

    1. 需要使用URLEncoder类先对中文名进行UTF-8编码,因为IE浏览器和谷歌浏览器收到含有 编码的字符串后会以UTF-8字符集进行解码显示
  2. 当浏览器是火狐浏览器:使用BASE64编解码

    1. BASE64编解码

代码实现:

DownLoad.java

  1. package com.chang.servlet;
  2. import org.apache.commons.io.IOUtils;
  3. import sun.misc.BASE64Encoder;
  4. import javax.servlet.ServletContext;
  5. import javax.servlet.ServletException;
  6. import javax.servlet.http.HttpServlet;
  7. import javax.servlet.http.HttpServletRequest;
  8. import javax.servlet.http.HttpServletResponse;
  9. import java.io.IOException;
  10. import java.io.InputStream;
  11. import java.io.OutputStream;
  12. import java.net.URLEncoder;
  13. public class DownLoad extends HttpServlet {
  14. @Override
  15. protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
  16. //1、获取要下载的文件名
  17. String downloadFileName = "2.png";
  18. //2、读取要下载的文件内容 (通过ServletContext对象可以读取)
  19. ServletContext servletContext = getServletContext();
  20. //3、获取要下载的文件类型
  21. String mimeType = servletContext.getMimeType("/file/" + downloadFileName);
  22. System.out.println("下载的文件类型:" + mimeType);
  23. //4、在回传前,通过响应头告诉客户端返回的数据类型
  24. resp.setContentType(mimeType);
  25. //5、还要告诉客户端收到的数据是用于下载使用(还是使用响应头)
  26. // Content-Disposition响应头,表示收到的数据怎么处理
  27. // attachment表示附件,表示下载使用
  28. // filename= 表示指定下载的文件名
  29. // url编码是把汉字转换成为%xx%xx的格式
  30. if (req.getHeader("User-Agent").contains("Firefox")) {
  31. // 如果是火狐浏览器使用Base64编码
  32. resp.setHeader("Content-Disposition", "attachment; filename==?UTF-8?B?" + new BASE64Encoder().encode("中国.jpg".getBytes("UTF-8")) + "?=");
  33. } else {
  34. // 如果不是火狐,是IE或谷歌,使用URL编码操作
  35. resp.setHeader("Content-Disposition", "attachment; filename=" + downloadFileName);
  36. // resp.setHeader("Content-Disposition", "attachment; filename=" + URLEncoder.encode("中国.jpg", "UTF-8"));
  37. }
  38. //只有后面三行代码可以实现在浏览器中获取请求的文件资源,因此需要前面的代码告知如何处理
  39. //斜杠被服务器解析表示地址为http://ip:port/工程名 映射 到代码的web目录
  40. InputStream resourceAsStream=servletContext.getResourceAsStream("/file/"+downloadFileName);
  41. //获取相应的输出流
  42. OutputStream outputStream= resp.getOutputStream();
  43. //使用commons-io这个包中的IOUtils实现读入与写出
  44. IOUtils.copy(resourceAsStream,outputStream);
  45. }
  46. }

web.xml

  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <web-app xmlns="http://xmlns.jcp.org/xml/ns/javaee"
  3. xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  4. xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/javaee http://xmlns.jcp.org/xml/ns/javaee/web-app_4_0.xsd"
  5. version="4.0">
  6. <servlet>
  7. <servlet-name>printHtml</servlet-name>
  8. <servlet-class>com.chang.servlet.PrintHtml</servlet-class>
  9. </servlet>
  10. <servlet-mapping>
  11. <servlet-name>printHtml</servlet-name>
  12. <url-pattern>/printHtml</url-pattern>
  13. </servlet-mapping>
  14. <servlet>
  15. <servlet-name>UploadServlet</servlet-name>
  16. <servlet-class>com.chang.servlet.UploadServlet</servlet-class>
  17. </servlet>
  18. <servlet-mapping>
  19. <servlet-name>UploadServlet</servlet-name>
  20. <url-pattern>/uploadServlet</url-pattern>
  21. </servlet-mapping>
  22. <servlet>
  23. <servlet-name>download</servlet-name>
  24. <servlet-class>com.chang.servlet.DownLoad</servlet-class>
  25. </servlet>
  26. <servlet-mapping>
  27. <servlet-name>download</servlet-name>
  28. <url-pattern>/download</url-pattern>
  29. </servlet-mapping>
  30. </web-app>