我的编程空间,编程开发者的网络收藏夹
学习永远不晚

Spring Boot实现文件上传的两种方式

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

Spring Boot实现文件上传的两种方式

最近的一个小项目里使用到了文件上传、下载功能,今天我打算梳理一下文件上传所涉及的技术及实现。 内容主要包括两部分,如何通过纯 Servlet 的形式进行文件上传、保存(不通过 Spring 框架);另一部分是如何在 Spring Web MVC 中进行文件上传。

01-从 HTTP 协议角度分析文件上传

HTTP 协议传输文件一般都遵循 RFC 1867 规范,即客户端通过 POST 请求,Context-Type 为 "multipart/form-data"。 前端提交页面一般为:

Choose a file:

通过 Wireshark 对 POST 请求进行抓包,发现发送的请求格式为:

POST /upload HTTP/1.1Host: localhost:8080Content-Length: 197624Content-Type: multipart/form-data; boundary=----WebKitFormBoundarynIbwtdWznj6QLu52First boundary: ------WebKitFormBoundarynIbwtdWznj6QLu52Encapsulated multipart part:  (image/png)    Content-Disposition: form-data; name="image"; filename="Snipaste_2023-01-05_13-35-11.png"    Content-Type: image/png    Portable Network GraphicsBoundary: ------WebKitFormBoundarynIbwtdWznj6QLu52Encapsulated multipart part:  (image/png)    Content-Disposition: form-data; name="image"; filename="Snipaste_2023-01-05_13-35-12.png"    Content-Type: image/png    Portable Network GraphicsLast boundary: ------WebKitFormBoundarynIbwtdWznj6QLu52--

对上述过程有了基本的理解后,就可以动手来写上传功能(本文以图片为例,当然你也可以实现支持上传其他类型的文件的版本)。 接下来我会展示两种实现文件上传功能的代码,第一种是使用纯 Servlet API 实现,不依赖 Spring 框架,当你的程序是一个简单的基于 Servlet 的应用时,可以参考这种方式。 第二种,借助了 Spring 提供的 MultipartFile 以及 MultipartResolver 实现的文件上传。

02-Servlet 处理上传请求

首先,需要先实现一个 Servlet。

@MultipartConfig(fileSizeThreshold = 5 * 1024 * 1024,        maxFileSize = 1024 * 1024 * 5,        maxRequestSize = 1024 * 1024 * 5)@WebServlet(name = "MultipartServlet", urlPatterns = "/servlet-upload")public class MultipartServlet extends HttpServlet {    private File uploadDir = null;    @Override    public void init(ServletConfig config) throws ServletException {        super.init(config);        // 检查存储文件的路径是否存在,若不存在,则创建一个        String uploadPath = System.getProperty("user.dir") + File.separator + "uploads";        uploadDir = new File(uploadPath);        if (!uploadDir.exists()) {            uploadDir.mkdir();        }    }    @Override    protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {        // 第一节中介绍过,文件上传是通过 POST 方法完成的,所以这里我们要重写 doPost 方法        try {            final Collection parts = req.getParts();   // 从请求中获取 multipart 内容            for (Part part : parts) {                if (part.getSize() <= 0) {                  // 判断上传的内容是否空文件                    System.out.println("part is empty, skip it!");                    continue;                }                String fileName = getFileName(part);       // 从请求中获取文件的名                // or                //final String fileName = part.getSubmittedFileName();                // fileName 是前端提供的,并不十分可靠                // 后端应该自己生成一个文件名                fileName = genNewFileName(fileName);                String uploadedFilePath = uploadDir + File.separator + fileName;                part.write(uploadedFilePath);   // 存储到指定目录                System.out.println("saved to " + uploadedFilePath);                resp.getWriter().write("saved to "  + uploadedFilePath);            }        } catch (ServletException se) {            // request is not of type multipart/form-data        }        resp.setStatus(HttpServletResponse.SC_OK);        resp.getWriter().flush();        resp.getWriter().close();    }    private String getFileName(Part part) {        for (String s : part.getHeader("Content-Disposition").split(";")) {            if (s.trim().startsWith("filename")) {                return s.substring(s.indexOf("=") + 2, s.length() - 1);            }        }        // 默认文件名        return "foo.txt";    }    private String genNewFileName(String filename) {        String filenameFormat = "%s.%s";        return String.format(filenameFormat,                UUID.randomUUID().toString().replace("-", "").substring(8),                FilenameUtils.getExtension(filename)        );    }}

这里面有几个地方需要解释一下;

  • 其一,getFileName 为什么要这么实现?参考第一节给出的 HTTP 报文,发现每个 Part,即两个 boundary 之间的内容,通过 Content-Disposition 给出了内容类型、文件名等信息。 getFileName 中的逻辑就是从这个格式里获得文件名的。 不过,这个文件名是由前端提供的,它其实也可以不提供,所以这个值就不是那么可靠。 所以,在我们将上传文件保存到磁盘上时,最好重新生成一个文件名,这就使 genNewFileName 的动机。
  • 其二,根据 HttpServletRequest 接口的文档,getParts 方法在请求不是 multipart 类型时会抛异常。 而且,Part 的内容有可能是为空的,如果我们不做判断,可能会在服务端创建一个空文件。
  • Servlet 类上的注解,@WebServlet 不再介绍,@MultipartConfig 是对请求、请求中文件大小的限制条件,当请求或文件超过这个限制时会抛对应的异常。

有了上面的定义,我们就可以测试下上传功能了。 

服务启动后,访问 页面能够得到上传页面。 选择文件,提交后,服务端响应成功,并将新名字传给前端。例如:

注:这里 会返回 Thymeleaf 实现的上传界面。

 @GetMapping("/servlet-upload-page")public String uploadImageByServlet(Model model) {    model.addAttribute("message", "please choose file to be uploaded");    return "upload/servlet-upload";}

界面内容为:

Upload Image Example

其中,@{/servlet-upload} 指向的是 @WebServlet(name = "MultipartServlet", urlPatterns = "/servlet-upload") 中将 Servlet 注册到的 url。

03-通过 Spring Boot 中的 MultipartFile 处理上传请求

通过 Spring Boot 来实现文件上传功能会更简单,它的自动化配置机制已经做了大部分的工作。 开发人员的工作就是定义一个 Controller,处理文件上传请求就可以了。

@Controllerpublic class UploadController {    public static String UPLOAD_DIRECTORY = System.getProperty("user.dir") + File.separator + "uploads";    @GetMapping("/upload")   // 主要返回文件上传页面    public String uploadImage(Model model) {        model.addAttribute("message", "please choose file to be uploaded");        return "upload/index";    }        @PostMapping("/upload")  // 处理文件上传 POST 请求    public String upload(@RequestParam("image")MultipartFile[] files,                         Model model)            throws IOException {        StringBuilder sb = new StringBuilder();        for (MultipartFile file : files) {            if (file.getSize() <= 0) {                continue;            }            final String newFileName = save(file);            final String msg = String.format("uploaded file %s, and new filename is %s%n", file.getOriginalFilename(), newFileName);            sb.append(msg);        }        model.addAttribute("msg", sb.toString());        return "upload/index";    }    private String save(MultipartFile file) throws IOException{        String newFileName = genNewFileName(file.getOriginalFilename());        final Path filePath = Paths.get(UPLOAD_DIRECTORY, newFileName);        Files.write(filePath, file.getBytes());        System.out.println("file saved to: " + filePath);        return newFileName;    }}

Spring Boot 中,文件上传请求(multipart request)被 StandardServletMultipartResolver 进一步封装为 StandardMultipartHttpServletRequest。 解析原请求的过程与我在前面介绍 Servlet 的方式时基本类似:

private void parseRequest(HttpServletRequest request) {    try {        Collection parts = request.getParts();        this.multipartParameterNames = new LinkedHashSet<>(parts.size());        MultiValueMap files = new LinkedMultiValueMap<>(parts.size());        for (Part part : parts) {              String headerValue = part.getHeader(HttpHeaders.CONTENT_DISPOSITION);            ContentDisposition disposition = ContentDisposition.parse(headerValue);            String filename = disposition.getFilename();            if (filename != null) {                // 把文件添加到 files                if (filename.startsWith("=?") && filename.endsWith("?=")) {                    filename = MimeDelegate.decode(filename);                }                // part 被封装为 StandardMultipartFile,它是 MultipartFile 的一个实现类                files.add(part.getName(), new StandardMultipartFile(part, filename));            }            else {                // 把不是文件的属性添加到 multipartParameterNames 中                this.multipartParameterNames.add(part.getName());            }        }        setMultipartFiles(files);    }    catch (Throwable ex) {        handleParseFailure(ex);    }}

通过上面的代码可以了解到,Client 提交的 POST 请求中,上传的文件被封装称 MultipartFile。 所以,我们在 Controller 中的处理方法中,可以通过 @RequestParam 的方式拿到这个文件列表进行处理,就像我们的 UploadController 实现的那样。

04-总结

在今天的文章中,我介绍了文件上传的两种实现方式,从纯 Servlet 实现,到基于 Spring Boot MVC 实现。 并且分析了 Spring Boot 中对 Multipart 请求的封装过程。

 

来源地址:https://blog.csdn.net/ww2651071028/article/details/129255059

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

Spring Boot实现文件上传的两种方式

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

Spring boot实现文件上传实例(多文件上传)

文件上传主要分以下几个步骤:(1)新建maven java project;(2)在pom.xml加入相应依赖;(3)新建一个表单页面(这里使用thymleaf);(4)编写controller;(5)测试;(6)对上传的文件做一些限制;(
2023-05-31

jsp 实现上传文件的两种方法

在用Java开发企业器系统的使用,特别是涉及到与办公相关的软件开发的时候,文件的上传是客户经常要提到的要求.因此有 一套很好文件上传的解决办法也能方便大家在这一块的开发.........[@more@]在用Java开发企业器系统的使用,特别
2023-06-03

Spring boot实现热部署的两种方式详解

热部署是什么大家都知道在项目开发过程中,常常会改动页面数据或者修改数据结构,为了显示改动效果,往往需要重启应用查看改变效果,其实就是重新编译生成了新的 Class 文件,这个文件里记录着和代码等对应的各种信息,然后 Class 文件将被虚拟
2023-05-31

spring boot ajax跨域的两种方式

前言 java语言在多数时,会作为一个后端语言,为前端的php,node.js等提供API接口。前端通过ajax请求去调用java的API服务。今天以node.js为例,介绍两种跨域方式:CrossOrigin和反向代理。 一、准备工作po
2023-05-31

SpringMVC上传文件的两种方法

在该示例中,阐述了SpringMVC如何上传文件。1、上传页面upload.jsp
2023-05-30

详解Spring-boot中读取config配置文件的两种方式

了解过spring-Boot这个技术的,应该知道Spring-Boot的核心配置文件application.properties,当然也可以通过注解自定义配置文件的信息。Spring-Boot读取配置文件的方式:一.读取核心配置文件信息ap
2023-05-31

vue实现上传按钮的样式的两种方法

这篇文章主要介绍了vue定制上传按钮的样式的两种方法,本文结合示例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
2022-12-24

java实现文件下载的两种方式

本文实例为大家分享了java实现文件下载的具体代码,供大家参考,具体内容如下public HttpServletResponse download(String path, HttpServletResponse response) {
2023-05-30

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录