19. web 应用常用功能 -文件上传下载


19.1 基本介绍

  1. 文件的上传和下载,是常见的功能
  2. 后面项目就使用了文件上传下载
  3. 如果是传输大文件,一般用专门工具或者插件
  4. 文件上传下载需要使用到两个包,需要导入
  5. 说明:

在这里插入图片描述

19.2 文件上传

19.2.1 文件上传的基本原理

● 文件上传原理示意图, 一图胜千言


在这里插入图片描述


19.2.2 文件上传应用实例

  • 需求说明:文件上传 应用实例如图

在这里插入图片描述


  • 思路分析/图解,看前面的原理示意图即可
  • 代码实现
  1. 创建 upload.jsp
<%--
  Created by IntelliJ IDEA.
  User: 谢家升
  Date: 2022/3/30
  Version: 1.0
--%>
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
    <!-- 指定了base标签 -->
    <base href="<%=request.getContextPath()+"/"%>>">
    <style type="text/css">
        input[type="submit"] {
            outline: none;
            border-radius: 5px;
            cursor: pointer;
            background-color: #31B0D5;
            border: none;
            width: 70px;
            height: 35px;
            font-size: 20px;
        }

        img {
            border-radius: 50%;
        }

        form {
            position: relative;
            width: 200px;
            height: 200px;
        }

        input[type="file"] {
            position: absolute;
            left: 0;
            top: 0;
            height: 200px;
            opacity: 0;
            cursor: pointer;
        }
    </style>

    <script type="text/javascript">
        function prev(event) {
            //获取展示图片的区域
            var img = document.getElementById("prevView");
            //获取文件对象
            var file = event.files[0];
            //获取文件阅读器: Js的一个类,直接使用即可
            var reader = new FileReader();
            reader.readAsDataURL(file);
            reader.onload = function () {
                //给img的src设置图片url
                img.setAttribute("src", this.result);
            }
        }
    </script>

</head>
<body>
<!-- 表单的enctype属性要设置为multipart/form-data
    enctype="multipart/form-data" 表示提交的数据是多个部分构造,有文件和文本
 -->
<form action="fileUploadServlet" method="post" enctype="multipart/form-data">
    家居图: <img src="2.jpg" alt="" width="200" height="200" id="prevView">
    <input type="file" name="pic" id="" value="" onchange="prev(this)"/>
    家居名: <input type="text" name="name"><br/>
    <input type="submit" value="上传"/>
</form>
</body>
</html>

  1. 创建 FileUploadServlet.java
package com.xjs.servlet;

import com.xjs.utils.WebUtils;
import org.apache.commons.fileupload.FileItem;
import org.apache.commons.fileupload.FileUploadException;
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.IOException;
import java.util.List;
import java.util.UUID;

/**
 * @Author: 谢家升
 * @Date: 2022/3/30-03-30-15:34
 * @Version: 1.0
 */
public class FileUploadServlet extends HttpServlet {
    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        System.out.println("FileUploadServlet 被调用...");

        //1.先判断是不是文件表单(enctype="multipart/form-data")
        if (ServletFileUpload.isMultipartContent(request)) {
            //System.out.println("ok");
            //2.创建 DiskFileItemFactory 对象,用于构建一个解析上传数据的工具对象
            DiskFileItemFactory diskFileItemFactory = new DiskFileItemFactory();
            //3.创建一个解析上传数据的工具对象
            /**
             *  表单提交的数据就是input元素
             *     家居图: <img src="2.jpg" alt="" width="200" height="200" id="prevView">
             *     <input type="file" name="pic" id="" value="" οnchange="prev(this)"/>
             *     家居名: <input type="text" name="name"><br/>
             *     <input type="submit" value="上传"/>
             */
            ServletFileUpload servletFileUpload = new ServletFileUpload(diskFileItemFactory);

            //解决接收到的文件名是中文乱码问题
            servletFileUpload.setHeaderEncoding("utf-8");

            //4.关键点, servletFileUpload 可用把表单提交的数据 text/文件,封装到 FileItem 文件项中
            //  编程技巧:如果我们不知道一个对象是什么结构【1.输出 2.debug 3.底层自动看到】
            try {
                List<FileItem> list = servletFileUpload.parseRequest(request);
                //System.out.println("list ==> " + list);
                /*
                list ==> [name=1.jpg, StoreLocation=D:\Projram\apache-tomcat-8.0.50\temp\xupload_38a26f60_17fd9b5f415__7f56_00000000.tmp, size=157734bytes, isFormField=false, FieldName=pic,
                          name=null, StoreLocation=D:\Projram\apache-tomcat-8.0.50\temp\xupload_38a26f60_17fd9b5f415__7f56_00000001.tmp, size=3bytes, isFormField=true, FieldName=name]

                 */

                //遍历list,并分别处理 ==> 感情的自然流露
                for (FileItem fileItem : list) {
                    //System.out.println("fileItem= " + fileItem);
                    //判断是不是一个文件 ==> 你是OOP程序员
                    if (fileItem.isFormField()) {//如果为true就是文本 input text
                        String name = fileItem.getString("utf-8");
                        System.out.println("家具名= " + name);
                    } else {//是一个文件
                        //获取上传的文件名
                        String name = fileItem.getName();
                        System.out.println("上传的文件名= " + name);

                        //把这个上传到 服务器的 temp目录下的文件保存到你指定的目录
                        //1.指定目录, 就是我们网站工作目录下
                        String filePath = "/upload/";
                        //2.获取到完整工作目录
                        String fileRealPath = request.getServletContext().getRealPath(filePath);
                        //System.out.println("fileRealPath= " + fileRealPath);
                        //fileRealPath= E:\javaweb_project\fileupdown\out\artifacts\fileupdown_war_exploded\xupload\

                        //3.创建这个上传的文件目录
                        //  思路:写一个工具类,可用返回 /2024/11/11 字符串
                        File fileRealPathDirectory = new File(fileRealPath + WebUtils.getYearMonthDay());
                        if (!fileRealPathDirectory.exists()) {//如果不存在就创建
                            fileRealPathDirectory.mkdirs();//创建
                        }

                        //4.将文件拷贝到 fileRealPathDirectory 目录下
                        //  构建一个上传文件的完整路径 : 目录+文件名
                        //  对上传的文件名进行处理,前面增加一个前缀,保证的唯一即可
                        name = UUID.randomUUID().toString() + "_" + System.currentTimeMillis() + "_" + name;
                        String fileFullPath = fileRealPathDirectory + "/" + name;
                        fileItem.write(new File(fileFullPath));

                        //5.提示信息
                        response.setContentType("text/html;charset=utf-8");
                        response.getWriter().write("<h1>上传成功~</h1>");
                    }
                }

            } catch (Exception e) {
                e.printStackTrace();
            }
        } else {
            System.out.println("不是文件表单...");
        }

    }

    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        doPost(request, response);
    }
}

  1. 创建工具类 webUtils.java
import java.time.LocalDateTime;

public class WebUtils {

    public static String getYearMonthDay() {
        //如何得到当前的日期-> java基础 日期 三代类
        LocalDateTime ldt = LocalDateTime.now();
        int year = ldt.getYear();
        int monthValue = ldt.getMonthValue();
        int dayOfMonth = ldt.getDayOfMonth();
        String yearMonthDay = year + "/" + monthValue + "/" + dayOfMonth + "/";
        return yearMonthDay;
    }
    //技巧, 最好测试一把
    //public static void main(String[] args) {
    //    System.out.println(WebUtils.getYearMonthDay());
    //}
}
  1. 测试看看文件是否上传成功

在这里插入图片描述


19.2.3 文件上传注意事项和细节

  1. 如果将文件都上传到一个目录下,当上传文件很多时,会造成访问文件速度变慢,因此可以将文件上传到不同目录 比如 一天上传的文件,统一放到一个文件夹 年月日,比如21001010 文件夹
  2. 一个完美的文件上传,要考虑的因素很多,比如断点续传、控制图片大小,尺寸,分片上传,防止恶意上传等,在项目中,可以考虑使用 WebUploader 组件(百度开发),链接: http://fex.baidu.com/webuploader/doc/index.html
  3. 文件上传功能,在项目中建议有限制的使用,一般用在头像、证明、合同、产品展示等,如果不加限制,会造成服务器空间被大量占用 [比如 b 站评论,就不能传图片,微信发1次朋友圈最多 9 张图等…]
  4. 文件上传,创建 web/upload 的文件夹,在 tomcat 启动时,没有在 out 目录下 创建 对应的 upload 文件夹, 原因是 tomcat 对应空目录是不会在 out 下创建相应目录的,所以,只需在 upload 目录下,放一个文件即可,这个是 Idea + Tomcat 的问题,实际开发不会存在 [演示]

19.3 文件下载

19.3.1 文件下载的原理分析图

● 文件下载原理分析图


在这里插入图片描述


19.3.2 文件下载应用实例

  • 需求:演示文件下载,如图

在这里插入图片描述


● 代码实现

  1. 先准备好可供下载的静态资源

在这里插入图片描述

  1. 创建 download.jsp
<%--
  Created by IntelliJ IDEA.
  User: 谢家升
  Date: 2022/3/30
  Version: 1.0
--%>
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>文件下载</title>
    <base href="<%=request.getContextPath()+"/"%>>">
</head>
<body>
<h1>文件下载</h1>
<a href="fileDownloadServlet?name=1.jpg">点击下载小狗图片</a><br/><br/>
<a href="fileDownloadServlet?name=高山流水.mp3">点击下载 高山流水.mp3</a><br/><br/>
</body>
</html>

  1. 创建 FileDownloadServlet.java
package com.xjs.servlet;

import org.apache.commons.io.IOUtils;
import sun.misc.BASE64Encoder;

import javax.servlet.ServletContext;
import javax.servlet.ServletException;
import javax.servlet.ServletOutputStream;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.io.InputStream;
import java.net.URLEncoder;

/**
 * @Author: 谢家升
 * @Date: 2022/3/30-03-30-18:24
 * @Version: 1.0
 */
public class FileDownloadServlet extends HttpServlet {
    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        //System.out.println("fileDownloadServlet 被调用...");
        //1.先准备要下载的文件【假定这些文件是公共的资源】
        //  重要提醒:保证当我们的Tomcat启动后,在工作目录out下有download文件夹,
        //  并且有可供下载的文件
        //  再次说明:如果没有看到你创建的download在工作目录out下,
        //  需要 rebuild project --> restart, 就 ok

        //2.获取到要下载的文件的名字
        request.setCharacterEncoding("utf-8");
        String downloadFileName = request.getParameter("name");
        System.out.println("downloadFileName= " + downloadFileName);

        //3.给http响应设置响应头 Content-Typ,就是 文件的MIME
        //  通过servletContext 来获取
        ServletContext servletContext = request.getServletContext();
        String downloadPath = "/download/"; //下载目录从web工程根目录计算 /download/1.jpg
        String downloadFileFullPath = downloadPath + downloadFileName;
        String mimeType = servletContext.getMimeType(downloadFileFullPath);
        System.out.println("mimeType= " + mimeType);
        response.setContentType(mimeType);

        //4.给http响应,设置响应头 Content-Disposition
        //  这里细节比较多,比如不同浏览器的写法不一样,还有要考虑编码
        //  ff 是 文件名中文需要 base64 编码 ,而 ie/chrome 是 URL 编码
        //  这里知道原理即可
        /*
            解读:
            (1) 如果是Firefox 则中文编码需要 base64
            (2) Content-Disposition 是指定下载的数据的展示形式 , 如果attachment 则使用文件下载方式
            (3) 如果是其他(主流ie/chrome) 中文编码使用URL编码
         */
        if (request.getHeader("User-Agent").contains("Firefox")) {
            // 火狐 Base64编码
            response.setHeader("Content-Disposition", "attachment; filename==?UTF-8?B?" +
                    new BASE64Encoder().encode(downloadFileName.getBytes("UTF-8")) + "?=");
        } else {
            // 其他(主流ie/chrome)使用URL编码操作
            response.setHeader("Content-Disposition", "attachment; filename=" +
                    URLEncoder.encode(downloadFileName, "UTF-8"));
        }

        //5.读取下载文件数据,返回给客户端/浏览器
        //(1) 创建一个和要下载的文件,关联的输入流
        InputStream resourceAsStream =
                servletContext.getResourceAsStream(downloadFileFullPath);
        //(2) 得到返回数据的输出流 【因为返回文件大多数是二进制(字节) 】
        ServletOutputStream outputStream = response.getOutputStream();

        //(3) 使用工具类,将输入流关联的文件,对拷到输出流,并返回给客户端/浏览器
        IOUtils.copy(resourceAsStream, outputStream);


    }

    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        doPost(request, response);
    }
}

  • 完成测试

在这里插入图片描述


19.3.3 文件下载注意事项和细节

  1. 文件下载,比较麻烦的就是文件名中文处理
  2. 因此,在代码中,针对不同浏览器做了处理

在这里插入图片描述

  1. 对于网站的文件,很多文件使用另存为即可下载,对于大文件(文档,视频),会使用专业的下载工具(迅雷、百度,腾讯,华为网盘等)
  2. 对于不同的浏览器,在把文件下载完毕后,处理的方式不一样,有些是直接打开文件,有些是将文件下载到 本地/下载目
  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
可以使用 Graham 扫描算法来解决这个问题。具体步骤如下: 1. 找到所有点中 y 坐标最小的点 P0,如果有多个点 y 坐标相同,则选择 x 坐标最小的点。 2. 将所有点按照与 P0 的极角从小到大排序,如果有多个点与 P0 的极角相同,则按照距离 P0 的距离从小到大排序。 3. 依次遍历排序后的点,对于每个点 Pi,如果 Pi、Pi-1 和 Pi-2 组成的向量不是逆时针方向,则将 Pi-1 从凸包中删除。 4. 最后得到的点集就是按照逆时针排列的凸包。 根据这个算法,可以得到这些点按照逆时针排列的结果为: 244.1821314.34-47.2585 243.6121314.59-54.2359 244.1481314.46-65.125 243.8691314.3-67.8689 242.9171313.88-71.8364 239.6851312.65-78.6564 236.711311.57-82.9158 233.3181310.38-86.6318 228.2821308.62-91.6409 223.9251307.12-94.9972 217.9131305.08-98.6861 209.4181302.27-101.409 201.7231299.79-101.566 194.1541299.61-19.8384 188.2361297.61-23.4317 180.3571294.85-31.5159 178.361294.12-34.3795 174.761292.76-41.8703 172.2181291.74-49.0167 170.9821291.2-54.0538 171.2971291.06-62.7229 172.4011291.28-67.9265 173.7171291.57-72.4188 174.4551291.75-74.6728 176.2711292.21-79.193 178.2681292.74-83.2374 179.8051293.16-85.9282 181.4961293.64-88.2235 186.3561295.05-94.0349 192.4441296.88-98.5715 201.3031301.97-18.1151 205.531303.33-18.1063 213.7331305.95-18.7536 218.3081307.39-20.0781 222.1911308.59-21.9997 225.9841309.74-24.5403 229.7841310.87-27.6483 232.0281311.54-29.8258 234.1151312.15-31.9929 237.1941313.03-35.9478 240.1011313.84-40.4844

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

要学就学灰太狼

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值