javaWeb学习-------文件下载及上传

文件上传及下载

​ 在web应用中,文件上传和下载功能是非常常用的功能。

1.文件下载

下载文件

  1. 要获取下载文件的路径
  2. 下载的文件名是啥?
  3. 设置想办法让浏览器能够支持下载我们需要的东西
  4. 获取下载文件的输入流
  5. 创建缓冲区
  6. 获取OutputStream对象
  7. 将FileOutputStream流写入到buffer缓冲区
  8. 使用OutputStream将缓冲区中的数据输出到客户端!
@Override 
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        // 1. 要获取下载文件的路径
        String realPath = "F:\\java\\javaweb- 02-servlet\\wang.png";
        System.out.println("下载文件的路径:"+realPath);
        //2.下载的文件名是啥?
        String fileName = realPath.substring(realPath.lastIndexOf("\\") + 1); //
        // 3. 设置想办法让浏览器能够支持(Content-Disposition)下载我们需要的东西,中文文件 名URLEncoder.encode编码,否则有可能乱码
        resp.setHeader("Content- Disposition","attachment;filename="+URLEncoder.encode(fileName,"UTF-8"));
        // 4. 获取下载文件的输入流
        FileInputStream in = new FileInputStream(realPath);
        // 5. 创建缓冲区
        int len = 0;
        byte[] buffer = new byte[1024];
        // 6. 获取OutputStream对象
        ServletOutputStream out = resp.getOutputStream();
        // 7. 将FileOutputStream流写入到buffer缓冲区,使用OutputStream将缓冲区中的数据 输出到客户端!
        while ((len=in.read(buffer))>0){
            out.write(buffer,0,len);
        }
        in.close();
        out.close(); 
    }

2.文件上传

创建一个空项目

创建web项目的三种方法:

①File–>New–>Module,在java中,选择web模块。

②在空项目中,右键选择项目选择Add Framework Support,选择框架

请添加图片描述
③在Maven中选择Web模板

1.文件传输原理及介绍

请添加图片描述

2.准备工作

对于文件上传,浏览器在上传的过程中是将文件以流的形式提交到服务器端的。

一般选择采用apache的开源工具common-fileupload这个文件上传组件。

common-fileupload是依赖于common-io这个包的,所以需要下载包:

在Maven资源库中下载jar包Maven Repository: Search/Browse/Explore (mvnrepository.com)

请添加图片描述

在项目中新建lib包,右键Add as Library添加到类库里,这样就可以查看源代码。

也可以通过打开java Structure查看结构。

请添加图片描述

【注意需要把导入的包添加到文件类库中】

请添加图片描述
文件上传依赖这两的包,这两个包的作用?

  • common-io
    • 通过一个io操作,把大量的工具类封装到这个包里边,拿来即用
    • 相当于网站的常用工具类打包,拿来即用
  • common-fileupload
    • 针对于文件上传,做的包

3.使用类介绍

  • 文件上传的注意事项:(面试问题,文件上传怎么调优)

1.为了保证服务器的安全,上传的文件应该放在外界无法访问到的目录下,比如放于Web-INF下;

2.为了防止文件覆盖的现象发生,要为上传的文件产生一个唯一的文件名;

​ 处理方法:①加后缀(时间戳)②-uuid 随机产生一串数字进行标识 ③-md5加密的算法 ④位运算算法

3.要限制文件上传的大小

4.可以限制文件上传的类型,在收到上传的文件时,判断后缀名是否合法

​ 比如:上传文件的类型——txt、doc

​ 上传图片的类型——jpg、png、bmp

  • 需要用到的类
ServletFileUpload:文件上传
Fileltem:文件遍历
DiskFileltemFactory:磁盘文件遍历一个迭代系统
fileltemFactory:文件工厂,

对应的设计模式:工厂模式
为了偷懒,不去new对象,用工厂生产对象
  • 流程

    ServletFileUpload负责处理上传的文件数据,并将表单中的每一个输入项封装成一个FileItme对象,在使用ServletFileUpload对象解析请求时需要DiskFileItmeFactory对象。所以,我们需要在进行解析工作前构造好DiskFileItmeFactory对象。通过ServletFile Upload 对象的构造方法或者setFileItmeFactory()方法设置ServletFileUpload对象的fileItmeFactory属性。

ServletFileUpload这个类专门负责:上传文件的数据,

ServletFileUpload这个类会把表单(上传文件的表单)中的每一项封装成一个对象Fileltem,

再使用ServletFileUpload对象(上传)解析请求时,需要DiskFileltemFactory对象。

我们再把它解析到磁盘的时候(放到网站,或服务器中),会需要它(ServletFileUpload)的构造方法,一个fileltemFactory属性

  • Fileltem类

在HTML页面input必须有name,在页面中必须有一个包含file的表单

让表单支持文件上传需要加属性,表单如果包含一个文件上传输入项的话,这个表单的enctype属性就必须设置为multipart/form-data

<form action="${pageContext.request.contextPath}/upload.do"enctype="multipart/form-data" method="post">
  上传用户:<input type="text" name="username"><br/>
  <p>上传文件1:<input type="file" name="file1"></p>
  <p>上传文件2:<input type="file" name="file2"></p>
  <%--submit:提交,reset:重置--%>
  <p><input type="submit" value="提交"> | <input type="reset"></p>
</form>

浏览器表单的类型如果为multipart/form-data,在服务器端想获取数据就要通过流。

【FileItme对象的常用方法介绍】

//isFormField方法用于:判断FileItem类对象封装的数据
//是一个普通文本表单还是一个文件表单,如果是普通表单字段则返回true,否则返回false
boolean isFormField();//判断表单的字段,没有文件的表单统称普通表单

//getFieldName方法用于:(获取)返回表单标签name属性的值。
string getFieldName();

//getstring方法用于将:FileItem对象中保存的数据流内容以一个字符串返回
string getstring();

//getName方法用于:获得文件上传字段中的文件名。
String getName();

//以流的形式返回上传文件的数据内容。
Inputstream getInputstream()

// delete方法用来:清空FileItem类对象中存放的主体内容
//如果主体内容被保存在临时文件中,delete方法将删除该临时文件。
void delete();
  • ServletFileUpload类

    ServletFileUpload:负责处理上传的文件数据,并将表单中每个输入项封装成一个Fileltem对象中,使用其parseRequest (HttpServletRequest)方法可以将通过表单中每一个HTML标签提交的数据封装成—个Fileltem对象,然后以List列表的形式

    返回。使用该方法处理上传文件简单易用。

4.代码实现

UpLoadFileServlet方法:

package com.wang.servlet;

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

import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.util.List;
import java.util.UUID;


public class UpLoadFileServlet extends HttpServlet {
    //处理表单
    @Override
    protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        //判断上传的文件是普通表单还是带文件的表单,两者有区别,文件的要传输,普通的表单收取字符串
        //这里判断的整个表单是不是multipart/form-data
        if (!ServletFileUpload.isMultipartContent(req)){
            return;//终止方法运行,说明这是一个普通表单
        }
        //创建文件上传需要保存路径,建议在WEB-INF路径下,安全,用户访问不到
        String uploadPath = this.getServletContext().getRealPath("/WEB-INF/upload");//得到servlet上下文的引用的真实路径
        //创建这个文件,并判断这个有没有
        File uploadFile = new File(uploadPath);
        if (!uploadFile.exists()){//如果这个文件没有则创建
            uploadFile.mkdir();
        }

        /*
        缓存:临时文件 tmp
        文件上传速度太慢了,给他个中间商,慢慢上传,保证是匀速进来的
         */
        //临时路径,假设文件超过了预期的大小,我们就把他放到一个临时文件中,过几天自动删除,或者提醒用户转存为永久
        String tmpPath = this.getServletContext().getRealPath("WEB-INF/tmp");//获得:全局上下文.真实地址
        //判断这个文件有没有
        File file = new File(tmpPath);
        if (!file.exists()) {//如果文件不存在
            file.mkdir();//创建这个临时目录
        }

        //处理上传的文件,一般都需要通过流来获取,我们可以使用request.getInputStream(),获取原生态的文件上传流,十分麻烦
        //但是我们都建议使用:Apache的文件上传组件来实现,common-fileupload,他需要依赖于 commons-io组件

         /*
        流程:ServletFileUpload负责处理上传的文件数据,并将表单中每个输入项封装成一个Fileltem对象,
        在使用ServletFilelpload对象(上传)解析请求时,需要DiskFileltemFactory对象。
        所以,我们需要在进行解析工作前构造好DiskFileltemFactory对象,
        通过SevletFileUpload对象的构造方法或setFileltemFactory()方法
        设置ServletFileUpload对象的fileltemFactory属性。
        */

        try {
            //1.创建 DiskFileItemFactory(磁盘工厂)对象,处理文件上传路径或者大小限制(给了磁盘空间)
            DiskFileItemFactory factory = getDiskFileItemFactory(file);
            //DiskFileItemFactory factory = new DiskFileItemFactory();

            //2.获取ServletFileUpLoad
            ServletFileUpload upload = getServletFileUpload(factory);
            //ServletFileUpload upload = new ServletFileUpload(factory); //(给了上传文件的输入流)
            //或者使用对象的方法进行设置:upload.setFileItemFactory(factory);

            //3.处理上传的文件
            String msg = uploadParseRequest(upload,req,uploadPath);

            //servlet请求转发消息
            req.setAttribute("msg", msg);//存
            req.getRequestDispatcher("info.jsp").forward(req, resp);

        } catch (FileUploadException e) {
            e.printStackTrace();
        }
    }

    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        super.doPost(req, resp);
    }

    public static DiskFileItemFactory getDiskFileItemFactory(File file) {
        DiskFileItemFactory factory = new DiskFileItemFactory();
        //通过这个工厂设置一个缓存区,当上传的文件大于这个缓冲区的时候,将他放入到临时文件中
        factory.setSizeThreshold(1024 * 1024);//缓冲区为1M
        factory.setRepository(file);//临时目录的保存目录,需要一个File
        return factory;
    }

    public static ServletFileUpload getServletFileUpload(DiskFileItemFactory factory) {
        ServletFileUpload upload = new ServletFileUpload(factory);
        //监听文件的上传进度
        upload.setProgressListener(new ProgressListener() {
            @Override
            //pBytesRead:已经读取到的文件大小
            //pContentLength:文件大小
            public void update(long pBytesRead, long pContentLength, int pItems) {
                System.out.println("总大小:" + pContentLength + "已上传:" + pBytesRead);
            }
        });
        //处理乱码问题
        upload.setHeaderEncoding("UTF-8");
        //设置单个文件的最大值
        upload.setFileSizeMax(1024 * 1024 * 10);
        //设置总共能够上传文件的大小
        //1024 = 1kb * 1024 = 1M * 10 = 10M
        upload.setSizeMax(1024 * 1024 * 10);
        return upload;
    }

    public static String uploadParseRequest(ServletFileUpload upload, HttpServletRequest request, String uploadPath) throws IOException, FileUploadException  {
        String msg = "";
        //3.把前端请求解析,封装成一个FileItem对象,需要从ServletFileUpload对象中获取
        List<FileItem> fileItems = upload.parseRequest(request);
        //fileItem 每一个表单对象
        for (FileItem fileItem : fileItems) {
            //判断上传的文件(控件)input是普通的表单text还是带文件的表单file
            if (fileItem.isFormField()) {//普通表单
                //getFieldName:前端表单控件的name
                String name = fileItem.getFieldName();
                String value = fileItem.getString("UTF-8");//处理乱码
                System.out.println(name + ":" + value);
            } else { //文件表单 ,(用到工具类)
                //===============处理文件===================
                //拿到文件名字
                String uploadFileName = fileItem.getName();
                System.out.println("上传的文件名:" + uploadFileName);
                //去重,如果有问题则判断下一个
                if (uploadFileName.trim().equals("") || uploadFileName == null) {
                    continue;
                }

                //获得上传的文件名  /images/girl/paojie/png
                //字符串游戏
                String fileName = uploadFileName.substring(uploadFileName.lastIndexOf("/") + 1);//最后一个 / +1
                //获得文件的后缀名
                String fileExtName = uploadFileName.substring(uploadFileName.lastIndexOf(".") + 1);
                 /*
                    如果文件后缀名 fileExtName 不是我们需要的,
                    就直接return,不处理,告诉用户文件类型不对。
                 */
                System.out.println("文件信息[件名:" + fileName + "---文件类型" + fileExtName + "]");

        /*面试题
          网络传输中的东西,都需要序列化
          POJO,实体类,如果想要在多个电脑上运行,  需要:传输==>把对象都序列化了
          写POJO实体类时,把这个接口 implements Serializable 加上 ,标记接口(没有方法的接口)
          标记接口:java虚拟机(JVM)在运行到的时候,它去识别,假设实现了 Serializable 这个接口,会帮你做一些事情。
          JVM中有一个本地方法栈:native,它是调用C++的,java虚拟机底层是C++写的。
          JNI = Java Native Interface  java本地化接口,
          java是无法操作计算机的,java是操作JVM的,JVM操作计算机(操作系统)
          JVM中还有一个java栈,平时写东西在java栈中写,本地方法栈是C++做的、操作系统来做
          公司里常用UUID,这个安全
        */
                //可以使用UUID(唯一识别的通用码),保证文件名唯一
                //UUID.randomUUID():随机生成一个唯一识别的通用码
                String uuidPath = UUID.randomUUID().toString();
                //===============处理文件完毕===================

                //==============文件存放地址完毕=================
                //文件存到哪? uploadPath
                //文件真实存在的路径 realPath
                String realPath = uploadPath + "/" + uuidPath;
                //给每个文件创建一个对应的文件夹
                File realPathFile = new File(realPath);
                if (!realPathFile.exists()) {//不存在
                    realPathFile.mkdir();//创建文件夹,保证不会重复
                }
                //===============存放地址完毕===================

                //获得文件上传的流
                InputStream inputStream = fileItem.getInputStream();

                //创建一个文件输出流
                FileOutputStream fos = new FileOutputStream(realPath + "/" + fileName);// 输出的地址

                //创建一个缓存区
                byte[] buffer = new byte[1024 * 1024];

                //判断是否读取完毕
                int len = 0;
                //如果大于0说明还存在数据
                while ((len = inputStream.read(buffer)) > 0) {
                    fos.write(buffer, 0, len);
                }
                //关闭流
                fos.close();
                inputStream.close();

                msg = "文件上传成功";
                fileItem.delete();//上传成功,清除临时文件
                //===============文件传输完毕===================

            }
        }
        return msg;

    }
}

web.xml文件

<servlet>
    <servlet-name>UpLoadFileServlet</servlet-name>
    <servlet-class>com.wang.servlet.UpLoadFileServlet</servlet-class>
</servlet>
<servlet-mapping>
    <servlet-name>UpLoadFileServlet</servlet-name>
    <url-pattern>/upload.do</url-pattern>
</servlet-mapping>

提交成功的表单

<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
    <title>Title</title>
</head>
<body>
${msg}
</body>
</html>

上传文件的地址可以通过Project Structure中找到:

请添加图片描述
请添加图片描述

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值