• [springboot专栏]文件本地上传与提供访问服务


    本章的核心内容是为大家介绍分布式文件系统,用于存储应用的图片、word、excel、pdf等文件。在开始介绍分布式文件系统之前,为大家介绍一下使用本机存储来存放文件资源。
    二者的核心实现过程是一样的:

    • 上传文件,保存文件(本节是本地磁盘)
    • 返回文件HTTP访问服务路径给前端,进行上传之后的效果展示

    一、复习

    服务端接收上传的目的是提供文件的访问服务,那么对于SpringBoot而言,有哪些可以提供文件访问的静态资源目录呢?

    • classpath:/META-INF/resources/ ,
    • classpath:/static/ ,
    • classpath:/public/ ,
    • classpath:/resources/

    这是之前的章节,我们为大家介绍的内容,从这里看出这里的静态资源都在classpath下。那么就出现问题:

    • 应用的文件资源不能和项目代码分开存储(你见过往github上传代码,还附带项目文件数据的么?)
    • 项目打包困难,当上传的文件越来越多,项目的打包jar越来越大。
    • 代码与文件数据不能分开存储,就意味着文件数据的备份将变得复杂

    二、文件上传目录自定义配置

    怎么解决上述问题?别忘记了spring boot 为我们提供了使用spring.resources.static-locations配置自定义静态文件的位置。

    web:
      upload-path: D:/data/
    
    spring:
      resources:
        static-locations: classpath:/META-INF/resources/,classpath:/resources/,classpath:/static/,classpath:/public/,file:${web.upload-path}
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 配置web.upload-path为与项目代码分离的静态资源路径,即:文件上传保存根路径
    • 配置spring.resources.static-locations,除了带上Spring Boot默认的静态资源路径之外,加上file:${web.upload-path}指向外部的文件资源上传路径。该路径下的静态资源可以直接对外提供HTTP访问服务。

    三、文件上传的Controller实现

    详情看代码注释

    @RestController
    public class FileUploadController {
    
      //绑定文件上传路径到uploadPath
      @Value("${web.upload-path}")
      private String uploadPath;
    
      SimpleDateFormat sdf = new SimpleDateFormat("yyyy/MM/dd/");
    
      @PostMapping("/upload")
      public String upload(MultipartFile uploadFile,
                           HttpServletRequest request) throws IOException {
    
        // 在 uploadPath 文件夹中通过日期对上传的文件归类保存
        // 比如:/2019/06/06/cf13891e-4b95-4000-81eb-b6d70ae44930.png
        String format = sdf.format(new Date());
        File folder = new File(uploadPath + format);
        if (!folder.isDirectory()) {
          folder.mkdirs();
        }
    
        // 对上传的文件重命名,避免文件重名
        String oldName = uploadFile.getOriginalFilename();
        String newName = UUID.randomUUID().toString()
                + oldName.substring(oldName.lastIndexOf("."), oldName.length());
    
    
        // 文件保存
        uploadFile.transferTo(new File(folder, newName));
    
        // 返回上传文件的访问路径
        //https://localhost:8888/2020/10/18/a9a05df4-6615-4bb5-b859-a3f9bf4bfae0.jpg
        String filePath = request.getScheme() + "://" + request.getServerName()
                + ":" + request.getServerPort() + "/"   + format + newName;
        return filePath;
      }
    
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36
    • 37
    • 38

    四、写一个模拟的文件上传页面,进行测试

    把该upload.html文件放到classpath:public目录下,对外提供访问。

    DOCTYPE html>
    <html lang="en">
    <head>
        <meta charset="UTF-8">
        <title>Titletitle>
    head>
    <body>
    <form action="/upload" method="post" enctype="multipart/form-data">
        <input type="file" name="uploadFile" value="请选择上传文件">
        <input type="submit" value="保存">
    form>
    body>
    html>
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13

    访问测试、点击“选择文件”,之后保存
    在这里插入图片描述

    文件被保存到服务端的web.upload-path指定的资源目录下
    在这里插入图片描述

    浏览器端响应结果如下,返回一个文件HTTP访问路径:
    在这里插入图片描述

    使用该HTTP访问路径,在浏览器端访问效果如下。证明我们的文件已经成功上传到服务端,以后需要访问该图片就通过这个HTTP URL就可以了。 或者使用前端html的img标签即可回显已上传图片.
    在这里插入图片描述
    欢迎关注公众号:字母哥杂谈,赠送各种我写的专栏PDF成书版本,如:回复003赠送专栏《docker修炼之道》。 字母哥博客:zimug.com

  • 相关阅读:
    基于Django搭建服务器平台的投票应用
    css笔记
    HCIP---IS-IS协议
    Pandas中的iloc函数; 查看.pt文件内容
    基于视觉识别的自动采摘机器人设计与实现
    有趣的设计模式——解救抓狂的商场收银员
    双系统时间问题、虚拟机扩展空间问题
    Uni-App常用事件
    如何做好一次代码审查,什么样是一次优秀的代码审查,静态代码分析工具有哪些
    线性代数基础概念:矩阵
  • 原文地址:https://blog.csdn.net/hanxiaotongtong/article/details/122906757