【Java】Web应用的文件上传下载

news2025/1/10 23:37:59

🎄欢迎来到@边境矢梦°的csdn博文🎄

 🎄本文主要梳理Web应用的文件上传下载🎄


🌈我是边境矢梦°,一个正在为秋招和算法竞赛做准备的学生🌈
🎆喜欢的朋友可以关注一下🫰🫰🫰,下次更新不迷路🎆

 Ps: 月亮越亮说明知识点越重要 (重要性或者难度越大)🌑🌒🌓🌔🌕

在这里插入图片描述

目录

📂 文件上传下载介绍🗂️

📂 第一章:上传轻松,让文件飞!🚀

📤基本原理(很细致, 也很易懂)

📂 第二章:下载无障碍,一键畅享!🔗

📥基本原理

📌MIME类型

📂 第三章:细节!🗂️


assorted files

📂 文件上传下载介绍🗂️

  1. 文件的上传和下载,是常见的功能。

  2. 后面项目就使用了文件上传下载。

  3. 如果是传输大文件,一般用专门工具或者插件

  4. 文件上传下载需要使用到两个包 , 需要导入

  5. 说明:

image-20220403171917107


📂 第一章:上传轻松,让文件飞!🚀

📤基本原理(很细致, 也很易懂)

注  : 其实主要就是DiskFileItemFactory  用于构建一个解析上传数据的工具对象

                              ServletFileUpload    创建一个解析上传数据的工具对象

                              List<FileItem>          解析器把表单提交的text/文件, 封装到 FileItem 文件项中  (多个FileItem构成的List)

  1. 客户端向服务器发送HTTP POST请求,其中包含要上传的文件数据。通常,这是通过HTML表单提交实现的。
  2. Tomcat服务器接收到该请求并将其路由到对应的Servlet。
  3. Servlet通过使用Apache Commons FileUpload库或Java Servlet API提供的multipart/form-data解析器来处理文件上传请求。
  4. 解析器会读取请求的内容,并将文件数据分解为多个部分
  5. 对于每个文件部分,解析器会创建一个FileItem对象该对象表示文件或表单字段。=== > 上传的文件的相关信息,如文件名、大小、内容等。
  6. 如果使用的是Apache Commons FileUpload库,通常会使用DiskFileItemFactory类来创建FileItem对象,该对象负责存储文件数据。
  7. 文件数据可以按照事先设定的大小阈值存储在内存中,或者超过阈值时被写入磁盘上的临时文件。这取决于DiskFileItemFactory的配置。
  8. Servlet通过遍历所有的FileItem对象来获取文件的相关信息,如文件名、大小、类型等,并将文件保存到指定的目录中。
  9. 在完成文件上传后,Servlet可以对上传的文件进行进一步操作,如文件处理、保存到数据库等。

 🌰示例 : 点击圆圈选择图片并输入文字上传到服务器

✍️ 代码 : 

@WebServlet(urlPatterns = {"/fileUploadServlet"})
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 元素
             *     <input type="file" name="pic" id="" value="2xxx.jpg" ο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 文件项中
            try {
                List<FileItem> list = servletFileUpload.parseRequest(request);
                    /*
                   list==>[
                   name=2022-04-04_131123.jpg, StoreLocation=D:\apache-tomcat-8.0.50\temp\xupload_1e5ae133_17ff39dae70__7c8c_00000000.tmp, size=30647bytes, isFormField=false, FieldName=pic,
                   name=null, StoreLocation=D:\apache-tomcat-8.0.50\temp\xupload_1e5ae133_17ff39dae70__7c8c_00000001.tmp, size=6bytes, isFormField=true, FieldName=name]
                     */
                System.out.println("list==>" + list);
                //遍历,并分别处理
                for (FileItem fileItem : list) {
                    System.out.println("fileItem=" + fileItem);
                    //判断是不是一个文件
                    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. 获取到完整目录 [io/servlet基础]
                        //  这个目录是和你的web项目运行环境绑定的. 是动态.
                        //fileRealPath=C:\ide\IdeaProjects\llp-javase\out\artifacts\fileupdown_war_exploded\xupload\
                        String fileRealPath =
                                request.getServletContext().getRealPath(filePath);
                        System.out.println("fileRealPath=" + fileRealPath);

                        //3. 创建这个上传的目录=> 创建目录?=> Java基础
                        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("上传成功~");

                    }
                }
        } catch (Exception e) {
            e.printStackTrace();
        }
    } else

    {
        System.out.println("不是文件表单...");
    }

}

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

📸 图片示意:上传成功

 


📂 第二章:下载无障碍,一键畅享!🔗

📥基本原理

  1. 客户端通过浏览器发送一个HTTP GET请求,该请求包含要下载文件的URL地址或其他必要的参数。
  2. Tomcat服务器接收到该请求并将其路由到对应的Servlet。
  3. Servlet根据请求中的参数或URL地址,确定要下载的文件的位置和名称。
  4. Servlet打开要下载的文件,并将其内容读取到一个输入流(例如,FileInputStream)中。
  5. Servlet设置响应的Content-Disposition头,以告诉浏览器将文件作为附件下载。可以使用response.setHeader(“Content-Disposition”, “attachment;filename=” + fileName)来设置。
  6. Servlet设置响应的Content-Type头,指定要下载的文件的MIME类型。例如,response.setContentType(“application/octet-stream”)指定下载的文件为二进制流。
  7. Servlet获取响应的输出流(例如,OutputStream)。
  8. Servlet使用输出流将文件内容写入到响应中,以便将文件发送到客户端。
  9. 客户端接收到相应的HTTP响应,并根据响应头中的Content-Disposition和Content-Type信息处理下载的文件。
  10. 浏览器将文件保存到指定的位置或显示下载对话框,供用户选择保存的位置。

📢 注意📢

文件下载的逻辑由Servlet来实现。Servlet在web.xml中进行配置,并通过相应的URL映射来处理文件下载请求。在Servlet中,需要设置正确的响应头,包括Content-Disposition和Content-Type,以确保文件能够正确地被下载和处理。

📌MIME类型

MIME(Multipurpose Internet Mail Extensions)类型是一种标准化的方法,用于表示文件的内容类型和格式。MIME类型是通过在HTTP协议中的Content-Type头部字段中进行指定的。它指示了服务器和接收者如何处理和解释传输的数据。

MIME类型由两部分组成:主类型和子类型,用斜杠(/)分隔。常见的MIME类型包括:

  1. text/plain:纯文本文件
  2. text/html:HTML文件
  3. text/css:CSS样式表文件
  4. application/json:JSON数据文件
  5. application/xml:XML文件
  6. application/pdf:PDF文件
  7. image/jpeg:JPEG图像文件
  8. image/png:PNG图像文件
  9. audio/mpeg:MP3音频文件
  10. video/mp4:MP4视频文件
  11. application/octet-stream:未知的二进制文件类型

这只是一些常见的MIME类型示例。实际上,MIME类型非常多,并且可以根据需要进行扩展。每种类型都有一个唯一的MIME类型,用于标识其特定的内容类型。

Web浏览器和服务器在处理文件时会根据文件的MIME类型执行相应的操作。例如,浏览器会根据MIME类型选择适当的插件或解析器来显示文件内容,而服务器则可以根据MIME类型确定如何处理和传输文件。

请注意,MIME类型由互联网工程任务组(IETF)指定和管理,并且可以在HTTP标准中找到完整的MIME类型列表。

 🌰示例 : 

1. 先准备要下载的文件[假定这些文件是公共的资源]

2. 获取到要下载的文件的名字

3. 给http响应,设置响应头 Content-Type , 就是文件的MIME通过servletContext 来获取

4. 给http响应,设置响应头 Content-Disposition

        (1)如果是Firefox 则中文编码需要 base64
        (2)Content-Disposition 是指定下载的数据的展示形式 , 如果attachment 则使用文件下载方式
        (3)如果是其他(主流ie/chrome) 中文编码使用URL编码

5. 读取下载的文件数据,返回给客户端/浏览器

        (1) 创建一个和要下载的文件,关联的输入流

        (2) 得到返回数据的输出流 [因为返回文件大多数是二进制(字节), IO java基础]

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

@WebServlet(urlPatterns = {"/fileDownLoadServlet"})
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-Type , 就是文件的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) 得到返回数据的输出流 [因为返回文件大多数是二进制(字节), IO java基础]
        ServletOutputStream outputStream = response.getOutputStream();

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

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

📸 图片示意:


 

📂 第三章:细节!🗂️

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

📸 图片示意:

🔗 : WebUploader 

🎉 这就是我为大家带来的关于Web应用文件上传下载的内容!通过简单的操作,上传和下载文件将变得轻松愉快,管理文件也将变得高效便利。快来尝试吧,你绝对不会后悔的!喜欢的话,就点个赞,分享给更多的小伙伴吧!👍

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/876412.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

日撸java三百行day84-87

文章目录 说明1. 网络结构与参数1.1 单层1.2 多层管理 2. CNN网络构建2.1 构建CNN网络建构2.2 初始化卷积神经网络FullCnn对象2.3 获取数据集2.4 训练数据2.4.1 前向传播函数forward2.4.1.1 CONVOLUTION卷积层2.4.1.2 SAMPLING 池化层2.4.1.3 OUTPUT 输出层 2.4.2 反向传播函数…

【华为Datacom 综合拓扑案例—分享篇】

拓扑图 题目要求 实验要求&#xff1a; 1、PC1\PC2\PC3\PC4采用DHCP自动获取IP地址&#xff0c;SW5作为服务器&#xff0c;SW3和SW4作为中继 创建地址池ip pool huawei1和ip pool huawei2&#xff0c;租期都为2天 2、SW3与SW4做链路聚合&#xff0c;采用LACP模式。SW3作为主…

tauri-react:快速开发跨平台软件的架子,支持自定义头部和窗口阴影效果

tauri-react 一个使用 taurireacttsantd 开发跨平台软件的模板&#xff0c;支持窗口头部自定义和窗口阴影&#xff0c;不用再自己做适配了&#xff0c;拿来即用&#xff0c;非常 nice。 开原地址&#xff1a;GitHub - Sjj1024/tauri-react: 一个最基础的使用tauri和react开发…

web测试与app测试的区别

web测试与app测试的区别 首先从系统架构来看的话&#xff1a; web项目&#xff0c;一般都是b/s架构&#xff0c;基于浏览器的&#xff0c;而app则是c/s的&#xff0c;必须要有客户端。那么在系统测试测试的时候就会产生区别了。 web测试只要更新了服务器端&#xff0c;客户端…

黑客利用 Facebook 漏洞,发起网络钓鱼攻击

Bleeping Computer 网站披露&#xff0c;网络攻击者利用 Salesforce 电子邮件服务和 SMTP 服务器中的漏洞&#xff0c;针对一些特定的 Facebook 账户发起复杂的网络钓鱼活动。 据悉&#xff0c;网络攻击者利用 Salesforce 等具有良好信誉的电子邮件网关分发网络钓鱼电子邮件&am…

YOLO v8目标跟踪详细解读(二)

上一篇&#xff0c;结合代码&#xff0c;我们详细的介绍了YOLOV8目标跟踪的Pipeline。大家应该对跟踪的流程有了大致的了解&#xff0c;下面我们将对跟踪中出现的卡尔曼滤波进行解读。 1.卡尔曼滤波器介绍 卡尔曼滤波&#xff08;kalman Filtering&#xff09;是一种利用线性…

网络安全(白帽黑客)大厂面试题

2023年过去了一大半&#xff0c;马上要到金九银十了 先来灵魂三连问&#xff0c;年初定的目标完成多少了&#xff1f;薪资涨了吗&#xff1f;女朋友找到了吗&#xff1f; ​好了&#xff0c;不扎大家的心了&#xff0c;接下来进入正文。 由于我之前写了不少网络安全技术相关的…

誉天HCIA-DataCom课程简介

总课时48小时&#xff0c;每天6课时&#xff0c;共8天 第一天 数据通信网络基础 1.区分网络通信和数据通信网络的概念 2.描述信息传递过程 3.区分不同网络设备&#xff0c;了解基本作用 4.认识不同网络类型以及拓扑类型 网络参考模型 1.理解数据的定义以及传递过程 2.理解网络…

精益制造的基础

一、相关概念 1、技术价值流 定义&#xff1a;把业务构想转化为向客户交付价值的、由技术驱动的服务所需要的流程 2、度量价值流性能指标 前置时间&#xff1a;工单创建后开始计时&#xff0c;到工作完成时结束 处理时间&#xff1a;从实际开始处理工作&#xff0c;到工作完成 …

智能工单系统是什么?它对企业后勤管理有什么作用?

智能工单系统是一种可以按照高度定制化的可视化流程进行设计和操作的工单系统&#xff0c;是报修、售后系统与工单系统的完美结合。与目前市场上的工单系统相比&#xff0c;的修智能工单系统对传统工单模板的操作方式进行了创新&#xff0c;而简单的工作台却没有失去任何工单功…

【从零学习python 】21.Python中的元组与字典

文章目录 元组一、访问元组二、修改元组三、count, index四、定义只有一个数据的元组五、交换两个变量的值 字典介绍一、列表的缺点二、字典的使用进阶案例 元组 Python的元组与列表类似&#xff0c;不同之处在于元组的元素不能修改。元组使用小括号&#xff0c;列表使用方括号…

亚马逊店铺推新品要注意什么?这些要点要牢记!

众所周知&#xff0c;对于一个亚马逊店铺而言&#xff0c;保持一定速度的产品更新换代是很有必要的&#xff0c;如果说产品不进行更新换代的话&#xff0c;随着消费者的需求发生改变&#xff0c;产品是很难有一个好销量的。 产品不断的更新换代&#xff0c;才能够更好的迎合消…

快速了解steam、csgo游戏搬砖,steam搬砖项目分享

科思创业汇 大家好&#xff0c;这里是科思创业汇&#xff0c;一个轻资产创业孵化平台。赚钱的方式有很多种&#xff0c;我希望在科思创业汇能够给你带来最快乐的那一种&#xff01; 我相信你对移动游戏并不陌生&#xff0c;但有些朋友不应该听说过steam和csgo这个词。steam是…

【javaweb】学习日记Day2 - JavaScript入门

目录 一、引入方式 1、内部脚本 2、外部脚本 二、基础语法 1、输出语句 2、定义变量类型 3、数据类型 4、运算符 &#xff08;1&#xff09;类型转换 5、函数 &#xff08;1&#xff09;方法一 &#xff08;2&#xff09;方法二 三、对象 1、Array数组 &#x…

自学stm32,需要会到什么程度能找到一份工作?

学STM32&#xff0c;想要找到一份工作&#xff0c;需要具备以下基本条件和技能&#xff1a;掌握新建工程和调试工程的基本操作&#xff0c;熟悉使用官方的STM32CubeIDE等开发工具。熟悉C语言编程&#xff0c;理解基本的语法和编程概念&#xff0c;对汇编语言有一定了解。熟悉ST…

国产商业漫画:题材、趋势与作者分析

国产商业漫画分析 本文是关于当前国产商业漫画市场的小研究&#xff0c;分析所用数据都是从漫画网站上直接爬取的。这里我选择的数据来源有两个&#xff0c;一个是当前收录商业化国漫最多的快看漫画&#xff0c;另一个是使用用户最多、以日漫为主的动漫之家。 在这篇分析里&a…

搭建openGauss 5.0 一主一从复制集群

openGauss是一款支持SQL2003标准语法&#xff0c;支持主备部署的高可用关系型国产数据库。 多种存储模式支持复合业务场景&#xff0c;新引入提供原地更新存储引擎。NUMA化数据结构支持高性能。Paxos一致性日志复制协议&#xff0c;主备模式&#xff0c;CRC校验支持高可用。支…

【Altium Designer】AD封装库+3D模型(2.95G)

【Altium Designer】AD封装库3D模型(2.95G) 如何添加3D封装库 https://blog.csdn.net/qq_42057393/article/details/115558858 组成 常用芯片封装 电阻 电容 电感 常用芯片 74系列芯片STC系列芯片ST系列芯片电源芯片通讯系列芯片未分类IC 晶体管 二极管三极管MOS管整流…

sql-libs靶场-----0x00、环境准备

文章目录 一、PhPstudy下载、安装二、Sqli-libs下载、搭建三、启用Sqli-libs phpstudy地址&#xff1a;https://www.xp.cn/ sqli-libs地址&#xff1a;https://github.com/Audi-1/sqli-labs 一、PhPstudy下载、安装 1、下载–解压–安装&#xff0c;安装完成如下图 2、更换php…

JS 删除的是最后一页的最后一条,页码设置逻辑

删除的场景&#xff1a; 解决思路&#xff1a; 1、计算操作后的总页数 2、删除成功之后的总页数与当前总页数进行比较 3、如果删除成功之后的总页数比小于当前总页数&#xff0c;需要把当前页码减去1&#xff1b;否则&#xff0c;直接进行列表数据的请求 代码实现 /*总条数…