创建名为spring_mvc_file的新module,过程参考9.1节和9.5节

11.1、文件下载

11.1.1、创建图片目录并放置图片

11.1.2、页面请求示例

<a th:href="@{/test/down}">下载图片</a>

11.1.3、控制器方法示例

package online.liaojy.controller;

import org.springframework.http.HttpHeaders;
import org.springframework.http.HttpStatus;
import org.springframework.http.ResponseEntity;
import org.springframework.stereotype.Controller;
import org.springframework.util.MultiValueMap;
import org.springframework.web.bind.annotation.RequestMapping; import javax.servlet.ServletContext;
import javax.servlet.http.HttpSession;
import java.io.*; /**
* @author liaojy
* @date 2023/11/2 - 20:50
*/
@Controller
public class FileDownloadController { // ResponseEntity 可以作为控制器方法的返回值,表示响应到浏览器的完整响应报文(包含响应体、响应体和响应状态码)
@RequestMapping("/test/down")
public ResponseEntity<byte[]> testResponseEntity(HttpSession session) throws IOException {
//获取ServletContext对象
ServletContext servletContext = session.getServletContext();
//获取服务器中工程的真实路径
String realPath = servletContext.getRealPath("");
//拼接工程路径和文件路径
//File.pathSeparator 表示文件路径分隔符,其能自动适应不同系统的路径分隔符(例如Windows的分隔符是\,Linux的分隔符是/)
realPath = realPath + File.separator + "img" + File.separator + "Windows.jpg";
//创建输入流
InputStream is = new FileInputStream(realPath);
//创建字节数组作为响应报文的响应体,is.available() 用于获取输入流所对应的文件的字节数
byte[] bytes = new byte[is.available()];
//将流读到字节数组中
is.read(bytes);
//创建HttpHeaders对象设置响应头信息
MultiValueMap<String, String> headers = new HttpHeaders();
//设置下载方式以及下载文件的名字
//请求头的键不区分大小写,Content Disposition头用来强制浏览器下载文件,而不是在浏览器中显示
//attachment表示以附件形式下载,即在下载时有提示信息
//filename表示下载的文件的名字
headers.add("Content-Disposition", "attachment;filename=Win10.jpg");
//设置响应状态码
HttpStatus statusCode = HttpStatus.OK;
//创建ResponseEntity对象
ResponseEntity<byte[]> responseEntity = new ResponseEntity<byte[]>(bytes, headers, statusCode);
//关闭输入流
is.close();
return responseEntity;
} }

11.1.4、测试效果

11.2、文件上传

11.2.1、页面请求示例

<!--
文件上传要求form表单的请求方式必须为post
并且设置属性enctype="multipart/form-data",表示表单中的数据以二进制的方式提交到服务器中
enctype属性值默认为application/x-www-form-urlencoded,传输不了文件
-->
<form th:action="@{/test/up}" method="post" enctype="multipart/form-data">
<!--
type属性值为file
name属性值要和控制器方法的形参一致
-->
图片:<input type="file" name="photo">
<br>
<input type="submit" value="上传图片">
</form>

11.2.2、引入依赖

        <!--文件上传-->
<dependency>
<groupId>commons-fileupload</groupId>
<artifactId>commons-fileupload</artifactId>
<version>1.3.1</version>
</dependency>

11.2.3、配置文件上传解析器

注意:springmvc 根据bean的id来获取文件解析器,因此必须设置 id 属性,且属性值必须为 multipartResolver

    <!--配置文件解析器-->
<!--必须通过文件解析器的解析才能将文件转换为 MultipartFile 对象-->
<!--文件解析器的id属性值必须为 multipartResolver -->
<bean id="multipartResolver" class="org.springframework.web.multipart.commons.CommonsMultipartResolver"> <!--文件上传最大值,默认值为 -1(表示没有限制)-->
<!--<property name="maxUploadSize" value="-1"></property>--> <!--解析请求的默认字符编码,默认值为"ISO-8859-1",通常设置为"UTF-8"-->
<!--<property name="defaultEncoding" value="UTF-8"></property>--> <!--存储在内存的最大值,默认值为10240B(10KB)-->
<!--<property name="maxInMemorySize" value="10240"></property>--> <!--每个文件上传最大值,默认值为 -1(表示没有限制)-->
<!--<property name="maxUploadSizePerFile" value="-1"></property>--> <!--延迟解析,默认为false(立即解析)-->
<!--<property name="resolveLazily" value="false"></property>--> <!--传文件的临时目录,默认值为WEB应用程序的临时目录-->
<!--<property name="uploadTempDir" value="classpath:online"></property>--> </bean>

11.2.4、控制器方法示例

package online.liaojy.controller;

import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.multipart.MultipartFile; import javax.servlet.ServletContext;
import javax.servlet.http.HttpSession;
import java.io.File;
import java.io.IOException; /**
* @author liaojy
* @date 2023/11/5 - 18:06
*/
@Controller
public class FileUploadController { @RequestMapping("/test/up")
public String testUp(MultipartFile photo, HttpSession session) throws IOException {
// 获取上传的文件的文件名
String originalFilename = photo.getOriginalFilename();
// 获取ServletContext对象
ServletContext servletContext = session.getServletContext();
// 获取服务器中当前工程下photo目录的真实路径
String photoPath = servletContext.getRealPath("photo");
// 创建photoPath所对应的file对象
File file = new File(photoPath);
// 判断file所对应目录是否存在
if (!file.exists()){
// file所对应目录不存在,则创建该目录
file.mkdir();
}
// 拼接目录路径和文件名为最终路径
String finalPath = photoPath + File.separator + originalFilename;
// 上传文件
photo.transferTo(new File(finalPath));
return "success";
} }

11.2.5、测试效果

11.3、文件上传的重名问题

11.3.1、问题描述

如果上传了同名的文件,那新的文件就会覆盖旧的文件;

因此需要通过一个机制,保证上传后的文件名的不会和新上传的文件名重复;

通常,可以使用uuid或时间戳,替换原来的文件名,从而保证文件名的唯一性。

11.3.2、解决方式

        // 获取上传的文件的后缀名
String suffixName = originalFilename.substring(originalFilename.lastIndexOf("."));
// 使用uuid生成的字符串和上传文件的后缀名,拼接成一个新的文件名
String fileName = UUID.randomUUID().toString() + suffixName;
        // 拼接目录路径和新的文件名为最终路径
String finalPath = photoPath + File.separator + fileName;

11.3.3、测试效果

11、SpringMVC之文件下载和上传的更多相关文章

  1. Pikachu漏洞练习平台实验——不安全的文件下载和上传(七)

    1.不安全的文件下载 1.1.概述 文件下载功能在很多web系统上都会出现,一般我们当点击下载链接,便会向后台发送一个下载请求,一般这个请求会包含一个需要下载的文件名称,后台在收到请求后 会开始执行下 ...

  2. pikachu-不安全的文件下载和上传

    一.文件下载漏洞 1.1 概述     很多网站都会提供文件下载的功能,即用户可以通过点击下载链接,下载到链接所对应的文件.但是,如果文件下载功能设计不当,则可能导致攻击者可以通过构造文件路径,从而获 ...

  3. idtcp实现文件下载和上传

    unit Unit1; interface uses Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls, Forms ...

  4. day58:Linux:BashShell&linux文件管理&linux文件下载上传

    目录 1.BashShell 2.Linux文件管理 3.Linux文件下载和上传 BashShell 1.什么是BeshShell? 命令的解释,用来翻译用户输入的指令 2.BashShell能做什 ...

  5. 使用springMVC实现文件上传和下载之文件下载

    接上一篇,文件下载需要获取下载文件的存储路径,这里只是手动填入,如果是在具体项目中,可以把文件名和上传后的存储路径保存在数据库中.然后增加一个文件列表的页面展示文件名和文件路径,然后点击下载的时候把相 ...

  6. [实战]MVC5+EF6+MySql企业网盘实战(12)——新建文件夹和上传文件

    写在前面 之前的上传文件的功能,只能上传到根目录,前两篇文章实现了新建文件夹的功能,则这里对上传文件的功能进行适配. 系列文章 [EF]vs15+ef6+mysql code first方式 [实战] ...

  7. web开发实战--图片裁剪和上传

    前言: 最近的开发中, 有一个上传头像的任务. 由于头像本身的特殊性, 其一般流程为选择图片, 编辑裁剪区域, 再继而上传图片操作. 看似简单的东西, 实则是挺麻烦的一件事. 借助这次开发机会, 来具 ...

  8. 基于uploadify.js实现多文件上传和上传进度条的显示

    uploadify是JQuery的一个插件,主要实现文件的异步上传功能,可以自定义文件大小限制.文件类型.是否自动上传等属性,可以显示上传的进度条.官网地址是http://www.uploadify. ...

  9. 使用FlashWavRecorder实现浏览器录制wav音频和上传音频文件,兼容IE8以上浏览器

    前言:本项目基于github开源插件实现,该插件使用flash实现,兼容IE8以上浏览器 感谢michalstocki的分享该项目,github项目地址:https://github.com/mich ...

  10. HTML5网页录音和上传到服务器,支持PC、Android,支持IOS微信

    准备做一个网页版聊天界面,表情啊.图片啊.上传文件啊都应该要有,视频就算了,语音还是要的. 本文记录的是在网页上用GitHub上的Recorder进行在线录音和上传到服务器,前几天升了一下级,以后有时 ...

随机推荐

  1. 第二次大作业BLOG心得

    (1)前言: 知识点: ①ArrayLsit的知识点: ArrayList 是 Java 中的动态数组实现,它提供了自动调整大小的功能,可以根据需要动态增长或收缩. ArrayList 可以存储任意类 ...

  2. Codeforces Round 923 (Div. 3) 比赛记录

    Codeforces Round 923 (Div. 3) 这是我第二次参加 cf阴间场. 10 minutes ago: 这次报名人数超过 4 万,一开始网站就崩溃了,比赛延迟了 10 分钟..开局 ...

  3. 支付宝spi接口设计验签和返回结果加签注意点,支付宝使用JSONObject对象

    支付宝spi接口设计验签和返回结果加签注意点,支付宝使用JSONObject对象 SPI 三方服务接入指南https://opendocs.alipay.com/isv/spiforisv 服务端实现 ...

  4. ESP8266串口WiFi模块 - WiFi杀手

    Tips:当你看到这个提示的时候,说明当前的文章是由原emlog博客系统搬迁至此的,文章发布时间已过于久远,编排和内容不一定完整,还请谅解` ESP8266串口WiFi模块 - WiFi杀手 日期:2 ...

  5. HTML5 在泛在电力物联网的 10 大业务领域 2/3D 可视化应用

    过去的 2018 年,我们认为是国内工业互联网可视化的元年,图扑软件作为在工业可视化领域的重度参与者,一线见证了众多 HTML5/Web 化.2D/3D 化的项目在工业界应用落地. 2019 年可以定 ...

  6. OB_MYSQL UPDATE 优化案例

    在工单系统上看到有一条SQL问题还没解决,直接联系这位同学看看是否需要帮忙. 慢SQL: UPDATE A SET CORPORATION_NAME = ( SELECT DISTINCT CORPO ...

  7. 推荐一个vs Nuget部署插件

    写在前面 nuget部署工具, 无论是直接用web上传还是用命令行工具上传,还是其他第三方工具我都没找到满意,直到那天在群里提了一下,有位大佬说了个NuPackvs插件,用了下,感觉基本满足了我的需求 ...

  8. 在WPF UWP WinUI中相同功能的用法的区别

    文件选择器 WPF:OpenFileDialog Microsoft.Win32.OpenFileDialog openFileDialog = new(); bool? result = openF ...

  9. STM32的内存管理(转)

    背景 这里针对STM32F407芯片+1M外部内存的内存管理!(全篇是个人愚见,如果错误,请不吝指出!) 定义 首先,定义3个内存池,分别是内部SRAM,外表SRAM和CCM:通过指定内存中的绝对地址 ...

  10. python基础-基本语句

    1 条件语句 在进行逻辑判断时,我们需要用到条件语句,Python 提供了 if.elif.else 来进行逻辑判断.格式如下所示: 1 if 判断条件1: 2 执行语句1... 3 elif 判断条 ...