JavaWeb文件上传

文件上传总览

文件上传主要是指将本地文件(包括但不限于图片、视频、音频等)上传到服务器,提供其他用户浏览或下载的过程。在日常生活中,我们在很多情况下都需要使用文件上传功能,比如:发微博、发朋友圈等(这里主要是指照片和视频文件),可见文件上传的重要性。文件上传功能实现需要涉及到两个部分,一个部分是前端程序、一个部分是后端程序,本文将前后端分开讲解。

接收文件

文件上传的前端准备

前端若想进行文件上传(网页的文件上传),就必须要有文件上传的三要素,前端必须要有这三要素才可以完成文件上传。前端要想实现文件上传就必须定义一个form表单,其中必须要有一个表单项类型为file;表单的提交方式必须是post方式,其原因是get方式提交有大小限制(并且不是很大),但文件上传可能涉及到大文件,所以说应该使用post方式提交表单;其最后一个要素,也是最重要的要素,就是在form表单中需要通过enctype属性设置表单的编码格式为 "multipart/form-data",这个设置非常重要——文件的本质上是二进制数据,而form表单的默认编码格式(x-www-form-urlencoded)是不适合传递二进制数据的。

假如说使用form表单默认的编码格式传输文件的问题:简单概括就是使用x-www-form-urlencoded编码格式传输文件,不会传输文件中的内容(数据),只能传输一个文件名。假如说我有一个.txt文本文件,使用默认的编码格式进行上传:

(该文件是有内容的)

该表单的编码格式是默认的x-www-form-urlencoded

 

发现上传的文件数据只有上传的文件的文件名,这证明了若使用form表单的默认编码格式,那么传输文件是失败的,只会上传文件的文件名。 总结一下文件上传的前端准备: 需要一个form表单,表单中需要一个表单项类型为file,并且该表单的提交方式必须为post,而且表单的编码格式必须修改为multipart/form-data。

文件上传的后端程序

文件上传的后端程序首先肯定是需要一个控制方法(SpringBoot),用于接收文件上传这个“功能接口”的请求,并且请求的其他数据之前怎么接收现在仍然怎么接收(比如简单参数、实体参数、日期参数等,还是像之前那么接收即可);请求提交的文件需要用SpringBoot提供的对象MultipartFile接收(前端中file表单项的名称必须和控制方法参数列表中MultipartyFile对象名完全相同)

创建请求处理类,并定义文件上传功能接口的控制器方法

@Slf4j
@RestController
public class FileController {

    @PostMapping("/upload")
    public Result upload(String username, Integer age, MultipartFile image) {
        log.info("文件上传:{},{},{}", username, age, image);
        return Result.success();
    }
}

此时通过前端页面发起请求:

<!DOCTYPE html>
<html lang="ch-ZN">
<head>
    <meta charset="UTF-8">
    <title>上传文件</title>
</head>
<body>

    <form action="/upload" method="post" enctype="multipart/form-data">
        姓名: <input type="text" name="username"><br>
        年龄: <input type="text" name="age"><br>
        头像: <input type="file" name="image"><br>
        <input type="submit" value="提交">
    </form>

</body>
</html>

 通过调试启动服务,发现此时可以成功接收到上传的文件,并获取文件的路径:

转到该路径下发现了三个.temp的临时文件: 

这是因为该表单有三个表单项,分别是两个text表单项(username和age)和一个file表单项(image文件),两个大小小的就是传的username和age的值,30kb那个temp文件就是上传的文件了,通过记事本打开验证一下:

 

 

现在已经在服务端接收到了文件,但是一但程序运行停止,这些临时文件也会消失——只要这次请求响应结束结束后(不管成功与否),临时文件都会被删除,所以说后端程序不但要接收上传的文件,更重要的是还需要将接收到的文件保存起来。总结一下文件上传的准备工作: 前端页面需要三要素:1.需要在form表单中定义一个表单项的type为file;2.需要指定表单的提交方式为post;3.需要指定表单的编码格式enctype为multipart/form-data。 这三要素缺一不可!后端需要使用Spring框架提供的MultipartFile类来接收前端上传的文件。

本地存储

上述代码成功在服务端接收到了客户端上传的文件,但是发现只能生成一个临时文件,一但这次请求响应结束后,.temp临时文件就会自动删除,这显然是不符合开发逻辑的,所以说我们需要将这些.temp临时文件存储起来。此处先介绍本地存储。

本地存储就是:在服务端接收到客户端上传的文件之后,将文件存储在本地服务器磁盘中。因为是基于SpringBoot框架,这其实并不困难,因为Spring框架中的MultipartFile对象提供了这样的方法——transferTo,transferTo的参数中需要传递一个File文件对象,就是将.temp临时文件写入到这个传递的文件对象中完成文件的本地存储。

此处在获取File对象的时候出现一个问题:不能把文件名写死了——因为假如有多个文件时,若把文件名写死了,那么后接收到的文件就会把前一次接收到的文件的内容覆盖了,整个功能接口就只能保存一个文件了。此时的解决方法是将File对象的文件名存储为上传文件的原文件名,MultpartFile对象中封装了上传文件的所有信息,并且提供了对应的方法获取这些信息,所以说直接可以通过操作MultipartFile对象得到原文件名:

@Slf4j
@RestController
public class FileController {

    @PostMapping("/upload")
    public Result upload(String username, Integer age, MultipartFile image) throws IOException {
        String originalFilename = image.getOriginalFilename();
        image.transferTo(new File("E:\imagesTest\" + originalFilename));
        return Result.success();
    }
}

此时代码就可以正确的接收到客户端上传的文件并将其保存在File指定的路径了:

 

但此时还有一个问题,假如说A用户上传了一个1.txt文件,接着B用户又上传了一个txt文件,也叫1.txt,那么就会出现B用户的文件内容将A用户的文件内容覆盖的问题,因为本地存储是按照文件的原名称存储的,遇到重名文件就会出现这样的问题。

避免文件名重复导致的覆盖问题

可能有人会想到使用System.currentTimeMillis()这个方法获取当前时间的毫秒值作为文件名避免重复,但是在一些大的项目中,有可能同一毫秒就正好有两个人上传了同一个同名的文件,这还是不能从根本上解决文件名重复的问题;解决这个问题可以使用UUID(通用唯一识别码)作为文件名进行存储,UUID是不会重复的(小概率事件不发生)。

但是又存在一个问题,假如说将UUID作为文件名,那么文件的扩展名是什么? 其实也非常简单,因为我们已经得到了文件的原名称,通过寻找到原文件名(字符串)中最后一个“.”所处的位置(因为无论文件名是什么,最后一个.之后的才是扩展名),找到最后一个“.”所处的位置之后,直接通过subString()方法截取到最后就可以获取文件的扩展名了,然后再将其和UUID进行拼接,就可以生成一个以UUID命名存储的文件了。

@PostMapping("/upload")
public Result upload(String username, Integer age, MultipartFile image) throws IOException {
    // 获取原文件名
    String originalFilename = image.getOriginalFilename();
    // 获取最后一个“.”的索引
    int index = originalFilename.lastIndexOf(".");
    // 通过subString方法和最后一个“.”索引进行截取,获取文件扩展名
    String externalName = originalFilename.substring(index);
    // UUID + 扩展名 = 新文件名
    String newFileName = UUID.randomUUID().toString() + externalName;
    image.transferTo(new File("E:\imagesTest\" + newFileName));
    return Result.success();
}

此时,文件在本地存储就是UUID了,再也不会有重复的可能了。

文件上传的大小配置

在SpringBoot框架中,文件上传,默认单个文件允许最大大小为1M。如果需要上传大文件,可以根据需求在application.properties(我使用的是yaml配置文件进行配置,思想是一样的,只是配置的格式不同)配置文件中进行设置:

servlet:
  multipart:
    # 配置单个文件最大上传大小(默认1MB)
    max-file-size: 10MB
    # 配置单个请求最大上传大小(一次请求可以上传多个文件)
    max-request-size: 100MB

这样配置之后,就可以根据需求上传大文件了。

本地存储不足

1.将文件存储到本地,无法通过浏览器等直接访问到文件。

2.若项目中上传了大量文件,这些文件都存储到服务器的本地磁盘,服务器的磁盘空间是十分宝贵的,用来存储用户上传的文件是十分浪费的,并且若服务器的磁盘存储空间满了,想要扩容也是十分困难的,并且现在很多项目都是服务器集群,扩容就更加困难了。

3.若服务器的磁盘损坏,存储的文件恢复极其困难,安全性不高。

现在几乎已经不使用本地存储了,都是使用云存储。

 

 

 

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Doris的基础架构

Doris的基础架构 Frontend&#xff08;FE&#xff09;&#xff1a;主要负责用户请求的接入、查询解析规划、元数据的管理、节点管理相关工作。Backend&#xff08;BE&#xff09;&#xff1a;主要负责数据存储、查询计划的执行。 我的Github地址&#xff0c;欢迎大家加入我的开…

Shell test 命令

Shell test 命令 Shell中的 test 命令用于检查某个条件是否成立&#xff0c;它可以进行数值、字符和文件三个方面的测试。 数值测试 参数说明-eq等于则为真-ne不等于则为真-gt大于则为真-ge大于等于则为真-lt小于则为真-le小于等于则为真 实例 num1100 num2100 if test $[n…

Kafka的消费消息是如何传递的?

大家好&#xff0c;我是锋哥。今天分享关于【Kafka的消费消息是如何传递的&#xff1f;】面试题。希望对大家有帮助&#xff1b; Kafka的消费消息是如何传递的&#xff1f; 1000道 互联网大厂Java工程师 精选面试题-Java资源分享网 Kafka 的消息传递是通过 消费者&#xff08…

Linux-ADC驱动实验

上一章我们讲解了如何给 ICM20608 编写 IIO 驱动&#xff0c;ICM20608 本质就是 ADC&#xff0c;因此纯粹的 ADC 驱动也是 IIO 驱动框架的。本章我们就来学习一下如何使用 I.MX6ULL 内部的 ADC&#xff0c;并且在学习巩固一下 IIO 驱动。 ADC 驱动源码简析 设备树下的 ADC 节点…

如何制作“优美”PPT

目录 1.免费PPT模板网站&#xff1a; 2.免费有较好质量的图片网站&#xff1a; 免费图片资源 免费透明PNG图片资源&#xff1a; 免费icon图片资源&#xff1a; 3.选择好的图片&#xff1a; 图片底色 4.要与不要 千万不要&#xff1a; 一定要&#xff1a; 6.一些建议…

SSRF对Redis进行内网渗透

SSRF对Redis进行内网渗透 一 环境搭建 准备一台服务器&#xff0c;开启lampp和redis&#xff0c;redis只允许内网访问 使用kali进行端口扫描&#xff0c;扫不到6379 使用kali连接redis&#xff0c;也连不上 ssrf漏洞代码 <?php ​$url $_GET[url]; ​// 创建一个cUR…

面经自测——死锁/死锁的必要条件/死锁的预防/进程通信的方式

前言 本文是作者专门用来自测Java后端相关面试题的&#xff0c;所有问题都是在牛客、知识星球或网上找到的最近最新的面试题&#xff0c;全文回答都是作者按自己的真实水平仿照真实环境的回答&#xff0c;所以答案不一定真实&#xff08;但回答一定真诚&#x1f923;&#xff0…

计算机网络研究实训室建设方案

一、概述 本方案旨在规划并实施一个先进的计算机网络研究实训室&#xff0c;旨在为学生提供一个深入学习、实践和研究网络技术的平台。实训室将集教学、实验、研究于一体&#xff0c;覆盖网络基础、网络架构、网络安全、网络管理等多个领域&#xff0c;以培养具备扎实理论基础…

React开发 - 技术细节汇总一

React简介 React 是一个声明式&#xff0c;高效且灵活的用于构建用户界面的 JavaScript 库。使用 React 可以将一些简短、独立的代码片段组合成复杂的 UI 界面&#xff0c;这些代码片段被称作“组件”。 ui render (data) -> 单向数据流 MVC // model var myapp {}; // …

嵌入式蓝桥杯学习4 lcd移植

cubemx配置 复制前面配置过的文件 打开cubemx&#xff0c;将PB8,PB9配置为GPIO-Output。 点击GENERATE CODE. 文件移植 1.打开比赛提供的文件包&#xff0c;点击Inc文件夹 2.点击Inc文件夹。复制fonts.h和lcd.h&#xff0c;粘贴到我们自己的工程文件夹的bsp中&#xff08…

迭代器模式的理解和实践

引言 在软件开发中&#xff0c;我们经常需要遍历容器对象&#xff08;如数组、列表、集合等&#xff09;中的元素。如果每个容器对象都实现自己的遍历算法&#xff0c;那么代码将会变得冗余且难以维护。为了解决这个问题&#xff0c;迭代器模式应运而生。迭代器模式是一种行为型…

STM32一keil5更换芯片后报错问题的解决。

目录 一、STM32型号认识二、报错问题三、常用的启动配置文件四、问题解决 一、STM32型号认识 二、报错问题 当我们在原来工程下修改芯片时&#xff0c;原本可以编译通过的代码突然很多报错。如下所示&#xff0c;这是因为我们的启动文件配置错误。对于不同型号的芯片其flash容量…

人工智能-自动驾驶领域

目录 引言自动驾驶与人工智能的结合为什么自动驾驶领域适合发表文章博雅智信的自动驾驶辅导服务结语 引言 自动驾驶技术的崛起是当代交通行业的一场革命。通过结合先进的人工智能算法、传感器技术与计算机视觉&#xff0c;自动驾驶不仅推动了技术的进步&#xff0c;也使得未来…

c++数据结构算法复习基础--11--高级排序算法-快速排序-归并排序-堆排序

高阶排序 1、快速排序 冒泡排序的升级算法 每次选择一个基准数&#xff0c;把小于基准数的放到基准数的左边&#xff0c;把大于基准数的放到基准数的右边&#xff0c;采用 “ 分治算法 ”处理剩余元素&#xff0c;直到整个序列变为有序序列。 最好和平均的复杂度&#xff1a…

修改MySQL存储路径

1.查看原路径 show variables like ‘%datadir%’; 2.停止MYSQL 以管理员身份运行命令提示符 net stop MySQL84 在服务中直接停止MySQL 3.编辑配置文件 可能会遇到无权限修改&#xff0c;可以先修改my.ini的权限。可以通过&#xff1a;右键my.ini → 属性 → 安全→ 编辑 …

微信小程序报错:http://159.75.169.224:7300不在以下 request 合法域名列表中,请参考文档

要解决此问题&#xff0c;需打开微信小程序开发者工具进行设置&#xff0c;打开详情-本地设置重新运行&#xff0c;该报错就没有啦

深入浅出:使用 Gin 框架生成 API 文档

深入浅出&#xff1a;使用 Gin 框架生成 API 文档 在现代 Web 开发中&#xff0c;API 文档是开发者之间沟通的重要桥梁。它不仅帮助前端开发者理解如何调用后端接口&#xff0c;还为测试人员和运维人员提供了宝贵的参考。对于 Go 语言开发者来说&#xff0c;Gin 是一个非常流行…

【 工具变量】IPCC碳排放因子数据测算表

一、数据简介&#xff1a; 排放因子法是IPCC提出的一种碳排放估算方法&#xff0c;也是目前适用范围最广、应用最为普遍的方法。将各类能源消耗的实物统计量转变为标准统计量&#xff0c;再乘以各自的碳排放因子&#xff0c;加总之后就可以得到碳排放总量。如果按照ISO14064标…

兔子的寿命有多长?

在宠物的世界里&#xff0c;兔子以其灵动的身姿、柔软的皮毛和温顺的性格深受人们喜爱。然而&#xff0c;当我们满心欢喜地将兔子迎进家门时&#xff0c;可曾想过它们能陪伴我们多久&#xff1f;兔子的寿命&#xff0c;是一个值得深入探讨的话题&#xff0c;它不仅关乎生命的时…

echarts地图立体效果,echarts地图点击事件,echarts地图自定义自定义tooltip

一.地图立体效果 方法1:两层地图叠加 实现原理:geo数组中放入两个地图对象,通过修改zlevel属性以及top,left,right,bottom形成视觉差 配置项参考如下代码: geo: [{zlevel: 2,top: 96,map: map,itemStyle: {color: #091A51ee,opacity: 1,borderWidth: 2,borderColor: #16BAFA…