Java SE 文件上传和文件下载的底层原理

1. Java SE 文件上传和文件下载的底层原理

@

目录
  • 1. Java SE 文件上传和文件下载的底层原理
  • 2. 文件上传
    • 2.1 文件上传应用实例
    • 2.2 文件上传注意事项和细节
  • 3. 文件下载
    • 3.1 文件下载应用实例
    • 3.2 文件下载注意事项和细节
  • 4. 总结:
  • 5. 最后:


2. 文件上传

  1. 文件的上传和下载,是常见的功能。说明:这里我们的文件上传仅仅只是对于小文件上的上传 。如果是传输大文件,一般用专门工具或者插件。

文件上传下载需要使用到如下两个包,需要导入。同时记得要进行导入加载到项目当中去。

在这里插入图片描述

在这里插入图片描述

文件上传原理示意图:

在这里插入图片描述

在这里插入图片描述

文件上传的解读:

  1. 文件上传还是使用的是表单 的方式提交
  2. 其中 action 还是按照以前规定来指定
  3. method 指定为 post ,因为文件上传是比较大的文件, get 无法发送较大的文件。
  4. enctype:encodetype 编码类型,要设置为:multipart/form-data ,表示进行二进制文件的提交,multipart/form-data: 表示表单提交的数据是有多个部分组成,也就是可以提交二进制数据和文本数据,两者都行。在这里插入图片描述
  1. 注意:enctype:encodetype 默认是:enctype="application/x-www-form-urlencoded" 即为 URL 编码,这种编码方式不适合对二进制文件数据的提交,一般适用于文本数据的提交。

操作上传文件流程:

  1. 判断是不是一个文件表单
  2. 判断表单提交的各个表单项是什么类型
  3. 如果是一个普通的表单项,就按照文本的方式来处理。
  4. 如果是一个文件表单项(二进制数据),使用 IO技术进行处理。
  5. 把表单提交的文件数据,保存到你指定的服务端的某个目录。

2.1 文件上传应用实例

在这里插入图片描述

对应文件上传的前端页面代码 : jsp

在这里插入图片描述

<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<!DOCTYPE html>
<html lang="en">
<head><meta charset="UTF-8"><title>Title</title><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");
//获取文件对象let file = event.files[0];
//获取文件阅读器let reader = new FileReader();reader.readAsDataURL(file);reader.onload = function () {
//给 img 的 src 设置图片 urlimg.setAttribute("src", this.result);}}</script>
</head>
<body>
<!-- 表单的 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="2xxx.jpg" onchange="prev(this)"/>家居名: <input type="text" name="name"><br/> <input type="submit" value="上传"/>
</form>
</body>
</html>

对应文件上传的Servlet 的编写。

在这里插入图片描述

package com.rainbowsea.servlet;import com.rainbowsea.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.nio.file.attribute.FileTime;
import java.util.List;
import java.util.UUID;public class FileUploadServlet extends HttpServlet {/*1. 判断是不是一个文件传单2. 判断表单提交的各个表单项是什么类型3. 如果是一个普通的表单项,就按照文本的方式来处理4. 如果是一个文件表单项(二进制数据),使用 IO技术进行处理5. 把表单提交的文件数据,保存到你指定的服务端的某个目录*/@Overrideprotected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException,IOException {//System.out.println("被调用了");// 1. 判断是不是 文件表单(enctype="multipart/form-data")if (ServletFileUpload.isMultipartContent(request)) {//System.out.println("OK");// 2. 创建 DiskFileItemFactory 对象,用于构建一个解析上传数据的工具对象DiskFileItemFactory diskFileItemFactory = new DiskFileItemFactory();// 3. 创建一个解析上传数据的工具对象ServletFileUpload servletFileUpload = new ServletFileUpload(diskFileItemFactory);// 4. 关键的地方 servletFileUpload 对象可以把表单提交的数据 text/ 文件// 将其封装到 FileItem 文件项中// 韩老师的编程心得体会:如果我们不知道一个对象是什么结构// 可以:1.输出该对象,2 debug 测试try {List<FileItem> list = servletFileUpload.parseRequest(request);//System.out.println("List ==>" + list);// 遍历,并分别处理=> 自然思路for (FileItem fileItem : list) {// java.lang.ClassCastException: org.apache.commons.fileupload.disk.DiskFileItem cannot be cast to java.nio.file.attribute.FileTime//System.out.println("fileItem == >" + fileItem);if (fileItem.isFormField()) { // 如果是 true 就是文本 input textString 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基础]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(); // 创建}// 解决接收到文件名是中文乱码问题servletFileUpload.setHeaderEncoding("utf-8");// 4. 将文件拷贝到 fileRealPathDirectory 目录// 构建一个上传文件的完整路径:目录 + 文件名// 有时-》上传失败了,可能是目录的问题 ,加上 “/”// 文本被替换覆盖的问题,我们也一个工具类,让文件名不重复// 对上传的文件名进行处理,前面增加一个前缀,保证是唯一即可name = UUID.randomUUID().toString() + "_" + System.currentTimeMillis() + "_" + name;String fileFullPath = fileRealPathDirectory + "/" + name;fileItem.write(new File(fileFullPath));// 提示信息response.setContentType("text/html;charset=utf-8");response.getWriter().write("上传成功");}}} catch (FileUploadException e) {throw new RuntimeException(e);} catch (Exception e) {throw new RuntimeException(e);}} else {System.out.println("不是文件表单...");}}}

上传文件操作的 Servlet 补充说明讲解:

有时-》上传失败了,可能是目录的问题 ,加上 “/”

为了防止大量的目录创建,可以增加日期时间进行创建多个目录,这样以日期天数进行创建目录的话,一年最多也就是 365个目录而已。

在这里插入图片描述

File fileRealPathDirectory = new File(fileRealPath+ WebUtils.getYearMonthDay());
String fileFullPath =  fileRealPathDirectory +"/"+ WebUtils.getYearMonthDay();public static String  getYearMonthDay() {// 如何得到当前的日期-》Java基础 日期,三代类LocalDateTime localDateTime = LocalDateTime.now();int year = localDateTime.getYear();int monthValue = localDateTime.getMonthValue();int dayOfMonth = localDateTime.getDayOfMonth();String yearMonthDay = year + "-" + monthValue + "-" + dayOfMonth;return yearMonthDay;}

文本被替换覆盖的问题,我们也一个工具类,让文件名不重复

// UUID.randomUUID().toString() 哈希不重复值
// System.currentTimeMillis() 获取当当前系统时间毫秒级别的
// 对上传的文件名进行处理,前面增加一个前缀,保证是唯一即可
// 同时使用特定的 "_" 符号进行分割,用于后续可能需要拿到文件名,最方便使用

name = UUID.randomUUID().toString() + "_" + System.currentTimeMillis() + "_" + name;
String fileFullPath = fileRealPathDirectory + "/" + name;

在这里插入图片描述

运行测试:看看文件是否能够上传成功

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述

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

  1. 如果将文件都上传到一个目录下,当上传文件很多时,会造成访问文件速度变慢,因此 可以将文件上传到不同目录 比如 一天上传的文件,统一放到一个文件夹 年月日, 比如:2024-7-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 的问题, 实际开发不会存 在

3. 文件下载

文件下载的原理分析图:

在这里插入图片描述

文件下载响应头说明:

  1. Content-Disposition: 表示下载的数据的展示方式,比如是内联形式(网页形式或者网页一部分)或者是文件下载方式 attachment
  2. Content-Type: 指定返回数据的类型 MIME ————》http 协议的内容

文件下载响应体说明:

  1. 在网络传输时是图片的原生数据(按照浏览器下载的编码)
  2. 这个图片时下载后查看到的,也就是浏览器本身做了解析

在这里插入图片描述

3.1 文件下载应用实例

对应文件上传的前端页面代码 : jsp

在这里插入图片描述

<%@ 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=java.png">点击下载Java图片</a><br/><br/>
<a href="fileDownLoadServlet?name=13-第十二章网络编程.pptx">点击下载 13-第十二章 网络编程.pptx</a><br/><br/>
</body>
</html>

注意:我们下载是,客户端从服务器端下载内容的了,所以我们需要模拟服务器,在服务器上添加上,我们客户端可以下载到的内容文件(这里:我们在 web 目录下,创建一个 download 目录,用于存放我们客户端(浏览器)可以下载到的文件)。

在这里插入图片描述

注意:创建好目录,添加好文件之后,要重新启动一下 Tomcat 服务器,让 这个我们添加的 download 资源目录,添加到 out 工作目录当中去。在这里插入图片描述

如果你重启了 Tomcat 服务器,也没有看到你创建的 download在工作目录 out下,则点击 rebuild project -> restart project

在这里插入图片描述

一个小细节:如果 web目录下创建的 目录是一个空文件夹/空目录,就是目录下没有东西的话,就算重启了 Tomcat 服务器也是不会添加到 out 目录下的。所以,只 需在 upload 目录下,放一个文件即可, 这个是 Idea + Tomcat 的问题, 实际开发不会存在

对应文件下载的Servlet 的编写。

在这里插入图片描述

package com.rainbowsea.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;public class FileDownLoadServlet extends HttpServlet {@Overrideprotected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {System.out.println("被调用");// 1. 先准备要下载的文件(假定这些文件时公共的资源)// 重要: 保证当我们的 tomcat 启动后,在工作目录下 有 out 有 download 文件夹,并且// 有可供下载的文件!!// 再次说明:如果你没有看到你创建的 download在工作目录 out下 rebuild project -> restart proj// 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/";  // 服务器资源图片,存放路径String downLoadFileFullPath = downLoadPath + downLoadFileName;String mimeType = servletContext.getMimeType(downLoadFileFullPath);System.out.println("mimeType = " + mimeType);response.setContentType(mimeType);// 4. 给http响应,设置响应头Content-Dispostion// 这里考虑的细节比较多,比如不同的浏览器写法不一样,考虑编码// ff 是文件名中文需要 base64, 而 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) 使用工具类,将输入流关联的文件,对拷到输出流,并返回给客户端/浏览器// 注意是: import org.apache.commons.io.IOUtils; 包下的IOUtils.copy(resourceAsStream,outputStream);}
}

上传下载操作的 Servlet 补充说明讲解:

文件下载,比较麻烦的就是不同浏览器文件名中文处理,因此,在代码中,需要针对不同的浏览器做处理。这里:火狐的 是文件名中文需要 base64 编码,而 ie/chrome 是 URL编码。针对不同的浏览器,我们需要进行不同的编码处理。

在这里插入图片描述

// 4. 给http响应,设置响应头Content-Dispostion// 这里考虑的细节比较多,比如不同的浏览器写法不一样,考虑编码// ff 是文件名中文需要 base64, 而 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"));}

运行测试:

在这里插入图片描述

在这里插入图片描述

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

  1. 文件下载,比较麻烦的就是不同浏览器文件名中文处理,因此,在代码中,需要针对不同的浏览器做处理。在这里插入图片描述

  2. 对于网站的文件,很多文件使用另存为即可下载,对于大文件(文档,视频),会使用专 业的下载工具(迅雷、百度,腾讯,华为网盘等)

  3. 对于不同的浏览器, 在把文件下载完毕后,处理的方式不一样, 有些是直接打开文件,有些是将文件下载到 本地/下载目录。

4. 总结:

  1. 文件上传的表单上的属性上的处理:method 指定为 post ,因为文件上传是比较大的文件, get 无法发送较大的文件。enctype:encodetype 编码类型,要设置为:multipart/form-data ,表示进行二进制文件的提交,multipart/form-data: 表示表单提交的数据是有多个部分组成,也就是可以提交二进制数据和文本数据,两者都行。

  2. 有时-》上传失败了,可能是目录的问题 ,加上 “/”

  3. 为了防止大量的目录创建,可以增加日期时间进行创建多个目录,这样以日期天数进行创建目录的话,一年最多也就是 365个目录而已。

  4. 文本被替换覆盖的问题,我们也一个工具类,让文件名不重复

    // UUID.randomUUID().toString() 哈希不重复值
    // System.currentTimeMillis() 获取当当前系统时间毫秒级别的
    // 对上传的文件名进行处理,前面增加一个前缀,保证是唯一即可
    // 同时使用特定的 "_" 符号进行分割,用于后续可能需要拿到文件名,最方便使用

    name = UUID.randomUUID().toString() + "_" + System.currentTimeMillis() + "_" + name;
    String fileFullPath = fileRealPathDirectory + "/" + name;
  5. 文件上传功能,在项目中建议有限制的使用,一般用在头像、证明、合同、产品展示等, 如果不加限制,会造成服务器空间被大量占用 [比如 b 站评论,就不能传图片,微信发 1 次朋友圈最多 9 张图等..]

  6. 文件下载:一个小细节:如果 web目录下创建的 目录是一个空文件夹/空目录,就是目录下没有东西的话,就算重启了 Tomcat 服务器也是不会添加到 out 目录下的。所以,只 需在 upload 目录下,放一个文件即可, 这个是 Idea + Tomcat 的问题, 实际开发不会存在。

  7. 文件下载,比较麻烦的就是不同浏览器文件名中文处理,因此,在代码中,需要针对不同的浏览器做处理。这里:火狐的 是文件名中文需要 base64 编码,而 ie/chrome 是 URL编码。针对不同的浏览器,我们需要进行不同的编码处理。

在这里插入图片描述

  1. 关于文件上传和下载,这里使用的是原生API的方式,在实际的开发中,我们这些关于文件上传和下载,都是被框架封装好了的,比如 :Spring MVC,Spring Boot 等等,我们只需要调用对应的API即可,框架封装的太好了,我们很难了解其中的底层原理。这里的文件上传和下载就是其底层原理了。

5. 最后:

限于自身水平,其中存在的错误,希望大家给予指教,韩信点兵——多多益善,谢谢大家,后会有期,江湖再见 !!!

在这里插入图片描述

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

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

使用Excel画出各类统计图(3)

本章会介绍如何画出箱线图 目录一、箱线图1.箱线图的作用2.箱线图的绘制与调整(1)画出箱线图(2)调整箱线图 一、箱线图 箱线图用于反映一组或多组连续型定量数据分布的中心位置和散布范围。 箱线图上需要体现出数据的某些特性,因此需要计算上四分位数、中位数、下四分位数…

Docker 和 k8s 学习

披个甲:偷的图灵学院的笔记 docker:https://note.youdao.com/ynoteshare/index.html?id=db5365c679b7d9129cbcfab5cb682d69&type=note&_time=1722071596141 k8s:https://note.youdao.com/ynoteshare/index.html?id=b2d5991b16e43cef9ac5071fbc516026&type=not…

帝国CMS如何设置是安全最优化的

帝国CMS如何设置是安全最优化的:(注:以下选项都是非必须设置,只是优化建议。)php配置文件php.ini设置:1、magic_quotes_gpc 设置为 On 魔术引用,此项建议开启。2、register_globals 设置为 Off PHP全局变量,此项建议关闭。3、display_errors 设置为 Off 不显示PHP错…

易优CMS内容页调用下载附件-模板代码使用说明{eyou:volist name=$eyou.field.file_list id=field} 附

{eyou:volist name="$eyou.field.file_list" id="field"}附件文件名:{$field.file_name}文件提取码:{$field.extract_code}服务器名称:{$field.server_name}文件大小:{$field.file_size}下载链接:{$field.file_url} {/eyou:volist}扫码添加技术【解决…

1251 - Client does not support authentication protocol requested by server; consider upgrading MySQL

错误记录: 1251 - Client does not support authentication protocol requested by server; consider upgrading MySQL client 错误原因: mysql8 之前的版本中加密规则是mysql_native_password,而在mysql8之后,加密规则是caching_sha2_password。 解决方案: 解决:①升级nav…

易优cms 安装常见问题汇总 Eyoucms快速入门

安装报错,请仔细核对数据库账号和密码答:请检查填写的数据库链接信息是否正确,此问题都是填写的数据库地址,账号,密码不正确导致的本地测试正常,放到虚拟主机就这样了安装的时候出现这个: 虚拟主机HPH5.5 ,数据库MYSQL 5.6 PHP Warning: include_once(./templates/step…

帝国CMS网站系统数据库配置文件是哪个文件?

帝国CMS系统数据库配置文件是哪个文件?/e/class/config.php扫码添加技术【解决问题,仅需10元起】专注中小企业网站建设、网站安全12年。熟悉各种CMS,精通PHP+MYSQL、HTML5、CSS3、Javascript等。承接:企业仿站、网站修改、网站改版、BUG修复、问题处理、二次开发、PSD转HTM…

【STC 相关】【转载】51单片机学习教程(简单入门)

转载自:知乎 https://zhuanlan.zhihu.com/p/628407258 作者:rakey (作者主页 https://www.zhihu.com/people/rakey-49)学习51单片机之前你一定要具备的基础知识。 1、C语言基础。 2、数字电路基础。 3、模拟电路基础。 如果你已经具备这些知识那么我们就可以来学习单片机。 …

【51系列】【转载】51单片机学习教程(简单入门)

转载自:知乎 https://zhuanlan.zhihu.com/p/628407258 作者:rakey (作者主页 https://www.zhihu.com/people/rakey-49)学习51单片机之前你一定要具备的基础知识。 1、C语言基础。 2、数字电路基础。 3、模拟电路基础。 如果你已经具备这些知识那么我们就可以来学习单片机。 …

2024暑假集训测试13

前言比赛链接。从来没见过交互题,T1 狂 CE 不止心态炸了,后面的题也没打好,T2、T3 简单题都不会了,所以为啥 T4 又放黑题。 T1 大众点评原题:AT_joisc2014_d。难点主要在交互,赛时琢磨了半场比赛终于搞明白是啥玩意儿了,可以将给定库当成压缩的一部分代码,可以调用里面…

51nod-3976-最长序列

https://class.51nod.com/Html/Textbook/ChapterIndex.html#textbookId=126&chapterId=338 https://class.51nod.com/Html/Textbook/Problem.html#problemId=3976&textbookChapterId=725 LIS是符号只有大于或小于,所以这道题就是LIS问题。 状态设计同LIS,由于答案就是…

Spring 常用的三种拦截器详解

在开发过程中,我们常常使用到拦截器来处理一些逻辑。最常用的三种拦截器分别是 AOP、 Interceptor 、 Filter,但其实很多人并不知道什么时候用AOP,什么时候用Interceptor,什么时候用Filter,也不知道其拦截顺序,内部原理。今天我们详细介绍一下这三种拦截器。前言 在开发过…