javajavaweb

一、文件的上传介绍

  1. 要有一个form标签,method = post请求方式,原因:文件的长度一般都会超过get请求的限制
  2. form标签的encType属性值必须为multipart/form-data
  3. 在form标签中使用input标签,type = file添加上传的文件
  4. 在form标签中使用input标签,type = submit提交到服务器
  5. 编写Servlet程序接收、处理上传的文件

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

在web目录下创建Upload.jsp

  1. <%@ page contentType="text/html;charset=UTF-8" language="java" %>
  2. <html>
  3. <head>
  4. <title>upload</title>
  5. </head>
  6. <body>
  7. <form action="http://localhost:8080/09_el_jstl/upload" method="post" enctype="multipart/form-data">
  8. <table>
  9. <tr>
  10. <td>用户名:</td>
  11. <td><input type="text" name="username"/></td>
  12. </tr>
  13. <tr>
  14. <td>头&nbsp;&nbsp;&nbsp;像:</td>
  15. <td><input type="file" name="photo"/></td>
  16. </tr>
  17. <tr>
  18. <td><input type="submit" value="上传"/></td>
  19. </tr>
  20. </table>
  21. </form>
  22. </body>
  23. </html>

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

创建Servlet程序upLoadServlet.java

首先导入两个jar包 (fileupload包依赖io包)
image.png

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

ServletFileUpload类,用于解析上传的数据
**public static final boolean isMultipartContent(HttpServletRequest request)**
如果上传的数据是多段的形式,返回true,只有多段的数据才是文件上传的
**public ServletFileUpload()**
空参构造器
**public ServletFileUpload(FileItemFactory fileItemFactory)**
参数为工厂实现类的构造器
**public List parseRequest(HttpServletRequest request)**
解析上传的数据,返回包含每一个表单项的List集合
FileItem类,表示每一个表单项
**public boolean isFormField()**
如果当前表单项是普通表单项,返回true,文件类型返回false
**public String getFieldName()**
获取当前表单项的name属性值
**public String getString()**
获取当前表单项的value属性值,参数为”UTF-8”可解决乱码问题
**public String getName()**
获取上传的文件名
**public void write(File file)**
将上传的文件写到参数File所指向的硬盘位置

import org.apache.commons.fileupload.FileItem;
import org.apache.commons.fileupload.FileItemFactory;
import org.apache.commons.fileupload.disk.DiskFileItemFactory;
import org.apache.commons.fileupload.servlet.ServletFileUpload;

import javax.servlet.*;
import javax.servlet.http.*;
import java.io.File;
import java.io.IOException;
import java.util.List;

public class UplaodServlet extends HttpServlet {

    /**
     * 用来处理上传的数据
     */
    @Override
    protected void doPost(HttpServletRequest request, HttpServletResponse response) 
        throws ServletException, IOException {
        //1 先判断上传的数据是否多段数据(只有是多段的数据,才是文件上传的)
        if (ServletFileUpload.isMultipartContent(request)) {

            // 创建FileItemFactory工厂实现类
            FileItemFactory fileItemFactory = new DiskFileItemFactory();
            // 创建用于解析上传数据的工具类ServletFileUpload类
            ServletFileUpload servletFileUpload = new ServletFileUpload(fileItemFactory);
            try {
                // 解析上传的数据,得到每一个表单项FileItem
                List<FileItem> list = servletFileUpload.parseRequest(request);

                // 循环判断,每一个表单项,是普通类型,还是上传的文件
                for (FileItem fileItem : list) {
                    if (fileItem.isFormField()) {
                        // 普通表单项
                        System.out.println("表单项的name属性值:" +         
                                           fileItem.getFieldName());
                        // 参数UTF-8.解决乱码问题
                        System.out.println("表单项的value属性值:" + 
                                           fileItem.getString("UTF-8"));

                    } else {
                        // 上传的文件
                        System.out.println("表单项的name属性值:" + 
                                           fileItem.getFieldName());
                        System.out.println("上传的文件名:" + fileItem.getName());
                        fileItem.write(new File("G:\\" + fileItem.getName()));
                    }
                }
            } catch (Exception e) {
                e.printStackTrace();
            }
        }
    }
}

在web.xml中编写servlet标签

<servlet>
        <servlet-name>UploadServlet</servlet-name>
        <servlet-class>com.atguigu.servlet.UplaodServlet</servlet-class>
</servlet>
<servlet-mapping>
        <servlet-name>UploadServlet</servlet-name>
        <url-pattern>/upload</url-pattern>
</servlet-mapping>

二、文件下载的过程

image.png

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

    文件下载过程详解

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

通过ServletContext的**getMimeType()**参数是要下载的文件所在路径,返回值是String类型

  1. 将获取的文件类型告诉客户端:

通过response的**setContentType()**参数是第二步的结果,无返回值

  1. 告诉客户端收到的数据用于下载使用(没有此步则内容直接显示在页面上):

通过response.**setHeader()**
参数是“Content-Disposition”, “attachment; fileName=xxx.xxx”

注意: Content-Disposition 响应头表示客户端收到的数据如何处理 attachment 表示附件,用于下载 filename 表示下载的文件名,可以与原文件名不同

  1. 获取要下载的文件并回传给客户端:

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

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

原因
response.setHeader(“Content-Disposition”, “attachment; fileName=中文名.jpg”);
如果下载的文件是中文名,会发现下载的文件无法正常显示汉字,原因是响应头中不能有汉字
解决

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

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

  1. 当浏览器是火狐浏览器:使用BASE64编解码

BASE64编解码

import sun.misc.BASE64Decoder;
import sun.misc.BASE64Encoder;

public class Base64Test {
    public static void main(String[] args) throws Exception {
        String content = "这是需要Base64编码的内容";

        // 创建一个Base64编码器
        BASE64Encoder base64Encoder = new BASE64Encoder();

        // 执行Base64编码操作
        String encodedString = base64Encoder.encode(content.getBytes("UTF-8"));
        System.out.println( encodedString );


        // 创建Base64解码器
        BASE64Decoder base64Decoder = new BASE64Decoder();

        // 解码操作
        byte[] bytes = base64Decoder.decodeBuffer(encodedString);
        String str = new String(bytes, "UTF-8");
        System.out.println(str);
    }
}

火狐浏览器下载文件乱码问题的解决
需要对中文名进行BASE64编码操作:
这时候需要把请求头 Content-Disposition: attachment; filename=中文名
编码成为 Content-Disposition: attachment; filename==?charset?B?xxxx?=
对 =?charset?B?xxxxx?= 的说明:

  • =? 表示编码内容的开始
  • charset 表示字符集(UTF-8、GBK等)
  • B 表示BASE64编码
  • xxxx 表示BASE64编码后的内容
  • ?= 表示编码内容的结束 ```java import org.apache.commons.io.IOUtils; import sun.misc.BASE64Encoder;

import javax.servlet.; import javax.servlet.http.; import java.io.IOException; import java.io.InputStream; import java.net.URLEncoder;

public class DownloadServlet extends HttpServlet {

@Override
protected void doGet(HttpServletRequest request, HttpServletResponse response) 
    throws ServletException, IOException {
    // 1、获取要下载的文件名
    String downloadFile = "1.jpg";

    // 2、通过ServletContext对象读取要下载的文件内容
    ServletContext servletContext = getServletContext();

    // 获取要下载的文件类型
    String mimeType = servletContext.getMimeType("/file/" + downloadFile);
    System.out.println("下载的文件类型" + mimeType);

    // 3、在回传前,通过响应头告诉客户端返回的数据类型
    response.setContentType(mimeType);

    // 4、还要告诉客户端收到的数据是用于下载使用(还是使用响应头)
    // Content-Disposition响应头,表示收到的数据怎么处理
    // attachment表示附件,表示下载使用
    // filename= 表示指定下载的文件名
    // url编码是把汉字转换成为%xx%xx的格式
    if (request.getParameter("User-Agent").contains("Firefox")) {
        // 如果是火狐浏览器,使用BASE64编码
        // =?charset?B?xxxxx?=
        // =?   表示编码内容的开始
        // charset  表示字符集
        // B        表示BASE64编码
        // xxxx     表示文件名BASE64编码后的内容
        // ?=   表示编码内容的结束
        response.setHeader("Content-Disposition", "attachment; filename==?UTF-8?B?"
                + new BASE64Encoder().encode("美女.jpg".getBytes("UTF-8")) + "?=");
    } else {
        // 如果不是火狐,是IE或谷歌,使用URL编码操作
        response.setHeader("Content-Disposition", "attachment; filename="
                + URLEncoder.encode("美女.jpg", "UTF-8"));
    }

    // 5、把下载的文件内容回传给客户端
    // 读取输入流中全部的数据,复制给输出流,输出给客户端
    InputStream resourceAsStream = servletContext.getResourceAsStream("/file/" 
                                   + downloadFile);
    ServletOutputStream outputStream = response.getOutputStream();
    IOUtils.copy(resourceAsStream, outputStream);
}

} ```

[

](https://blog.csdn.net/weixin_49343190/article/details/108249513)