Loading [MathJax]/jax/output/CommonHTML/config.js
首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >专栏 >SpringBoot整合Minio对象存储服务

SpringBoot整合Minio对象存储服务

原创
作者头像
甄士隐
修改于 2022-01-27 00:44:48
修改于 2022-01-27 00:44:48
2.1K0
举报
文章被收录于专栏:混合云混合云

MinIO 是一款高性能、分布式的对象存储系统. 它是一款软件产品, 可以100%的运行在标准硬件。即X86等低成本机器也能够很好的运行MinIO。

MinIO与传统的存储和其他的对象存储不同的是:它一开始就针对性能要求更高的私有云标准进行软件架构设计。因为MinIO一开始就只为对象存储而设计。所以他采用了更易用的方式进行设计,它能实现对象存储所需要的全部功能,在性能上也更加强劲,它不会为了更多的业务功能而妥协,失去MinIO的易用性、高效性。 这样的结果所带来的好处是:它能够更简单的实现局有弹性伸缩能力的原生对象存储服务。

MinIO在传统对象存储用例(例如辅助存储,灾难恢复和归档)方面表现出色。同时,它在机器学习大数据、私有云、混合云等方面的存储技术上也独树一帜。当然,也不排除数据分析、高性能应用负载、原生云的支持。

1.添加依赖

代码语言:html
AI代码解释
复制
<!--minio-->
<dependency>
    <groupId>io.minio</groupId>
    <artifactId>minio</artifactId>
    <version>8.2.1</version>
</dependency>

2.添加配置

代码语言:yaml
AI代码解释
复制
spring:
  # 配合minio加大内存限制
  servlet:
    multipart:
      max-file-size: 5120MB
      max-request-size: 5120MB
  # minio对象存储服务配置
  minio:
    url: www.jyglxt.com:9010
    access-key: caixibei
    secret-key: Abcd123.
    bucket: software

2.编写配置类

代码语言:java
AI代码解释
复制
package cn.jxd.caixibei.config;
import cn.jxd.caixibei.units.MinioUtils;
import lombok.extern.slf4j.Slf4j;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
@Slf4j
@Configuration
public class MinioConfig {
    /**
     * url
     */
    @Value(value = "${spring.minio.url}")
    private String minioUrl;

    /**
     * username
     */
    @Value(value = "${spring.minio.access-key}")
    private String minioName;

    /**
     * password
     */
    @Value(value = "${spring.minio.secret-key}")
    private String minioPass;

    @Value(value = "${spring.minio.bucket}")
    private String bucketName;

    @Bean
    public void initMinio(){
        if(!minioUrl.startsWith("http")){
            minioUrl = "http://" + minioUrl;
        }
        if(!minioUrl.endsWith("/")){
            minioUrl = minioUrl.concat("/");
        }
        MinioUtils.setMinioUrl(minioUrl);
        MinioUtils.setMinioName(minioName);
        MinioUtils.setMinioPass(minioPass);
        MinioUtils.setBucketName(bucketName);
    }
}

3.编写工具类

代码语言:java
AI代码解释
复制
package cn.jxd.caixibei.units;
import io.minio.*;
import io.minio.errors.*;
import io.minio.http.Method;
import io.minio.messages.Item;
import lombok.extern.slf4j.Slf4j;
import org.springframework.web.multipart.MultipartFile;
import java.io.IOException;
import java.io.InputStream;
import java.net.URLDecoder;
import java.security.InvalidKeyException;
import java.security.NoSuchAlgorithmException;
import java.time.format.DateTimeFormatter;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
import java.util.regex.PatternSyntaxException;
/**
 * minio文件上传工具类
 * @author caixibei
 */
@Slf4j
public class MinioUtils {
    private static String minioUrl;
    private static String minioName;
    private static String minioPass;
    private static String bucketName;

    public static void setMinioUrl(String minioUrl) {
        MinioUtils.minioUrl = minioUrl;
    }

    public static void setMinioName(String minioName) {
        MinioUtils.minioName = minioName;
    }

    public static void setMinioPass(String minioPass) {
        MinioUtils.minioPass = minioPass;
    }

    public static void setBucketName(String bucketName) {
        MinioUtils.bucketName = bucketName;
    }

    private static MinioClient minioClient = null;

    /**
     * 判断文件名是否带盘符,重新处理
     * @param fileName
     * @return
     */
    public static String getFileName(String fileName){
        //判断是否带有盘符信息
        int unixSep = fileName.lastIndexOf('/');
        int winSep = fileName.lastIndexOf('\\');
        int pos = (winSep > unixSep ? winSep : unixSep);
        if (pos != -1)  {
            fileName = fileName.substring(pos + 1);
        }
        //替换上传文件名字的特殊字符
        fileName = fileName.replace("=","").replace(",","").replace("&","")
                .replace("#", "").replace("“", "").replace("”", "");
        //替换上传文件名字中的空格
        fileName=fileName.replaceAll("\\s","");
        return fileName;
    }

    /**
     * 清除掉所有特殊字符
     * @param str
     * @return
     * @throws PatternSyntaxException
     */
    public static String filter(String str) throws PatternSyntaxException {
        // 清除掉所有特殊字符
        String regEx = "[`_《》~!@#$%^&*()+=|{}':;',\\[\\].<>?~!@#¥%……&*()——+|{}【】‘;:”“’。,、?]";
        Pattern p = Pattern.compile(regEx);
        Matcher m = p.matcher(str);
        return m.replaceAll("").trim();
    }

    /**
     * 上传文件
     * @param file
     * @return
     */
    public static String upload(MultipartFile file, String bizPath, String customBucket) {
        String file_url = "";
        bizPath= filter(bizPath);
        String newBucket = bucketName;
        if(customBucket!=null){
            newBucket = customBucket;
        }
        try {
            initMinio(minioUrl, minioName,minioPass);
            // 检查存储桶是否已经存在
            if(minioClient.bucketExists(BucketExistsArgs.builder().bucket(newBucket).build())) {
                log.info("Bucket already exists.");
            } else {
                // 创建一个名为ota的存储桶
                minioClient.makeBucket(MakeBucketArgs.builder().bucket(newBucket).build());
                log.info("create a new bucket.");
            }
            InputStream stream = file.getInputStream();
            // 获取文件名
            String orgName = file.getOriginalFilename();
            if("".equals(orgName)){
                orgName=file.getName();
            }
            orgName = getFileName(orgName);
            String objectName = bizPath+"/"
                    +( orgName.indexOf(".")==-1
                    ?orgName + "_" + System.currentTimeMillis()
                    :orgName.substring(0, orgName.lastIndexOf(".")) + "_" + System.currentTimeMillis() + orgName.substring(orgName.lastIndexOf("."))
            );
            // 使用putObject上传一个本地文件到存储桶中。
            if(objectName.startsWith("/")){
                objectName = objectName.substring(1);
            }
            PutObjectArgs objectArgs = PutObjectArgs.builder().object(objectName)
                    .bucket(newBucket)
                    .contentType("application/octet-stream")
                    .stream(stream,stream.available(),-1).build();
            minioClient.putObject(objectArgs);
            stream.close();
            file_url = minioUrl+newBucket+"/"+objectName;
        }catch (Exception e){
            log.error(e.getMessage(), e);
        }
        return file_url;
    }

    /**
     * 文件上传
     * @param file
     * @param bizPath
     * @return
     */
    public static String upload(MultipartFile file, String bizPath) {
        return  upload(file,bizPath,null);
    }

    /**
     * 获取文件流
     * @param bucketName
     * @param objectName
     * @return
     */
    public static InputStream getMinioFile(String bucketName,String objectName){
        InputStream inputStream = null;
        try {
            initMinio(minioUrl, minioName, minioPass);
            GetObjectArgs objectArgs = GetObjectArgs.builder().object(objectName)
                    .bucket(bucketName).build();
            inputStream = minioClient.getObject(objectArgs);
        } catch (Exception e) {
            log.info("文件获取失败" + e.getMessage());
        }
        return inputStream;
    }

    /**
     * 删除文件
     * @param bucketName
     * @param objectName
     * @throws Exception
     */
    public static void removeObject(String bucketName, String objectName) {
        try {
            initMinio(minioUrl, minioName,minioPass);
            RemoveObjectArgs objectArgs = RemoveObjectArgs.builder().object(objectName)
                    .bucket(bucketName).build();
            minioClient.removeObject(objectArgs);
        }catch (Exception e){
            log.info("文件删除失败" + e.getMessage());
        }
    }

    /**
     * 获取文件外链
     * @param bucketName
     * @param objectName
     * @param expires
     * @return
     */
    public static String getObjectURL(String bucketName, String objectName, Integer expires) {
        initMinio(minioUrl, minioName,minioPass);
        try{
            GetPresignedObjectUrlArgs objectArgs = GetPresignedObjectUrlArgs.builder().object(objectName)
                    .bucket(bucketName)
                    .expiry(expires).build();
            String url = minioClient.getPresignedObjectUrl(objectArgs);
            return URLDecoder.decode(url,"UTF-8");
        }catch (Exception e){
            log.info("文件路径获取失败" + e.getMessage());
        }
        return null;
    }

    /**
     * 初始化客户端
     * @param minioUrl
     * @param minioName
     * @param minioPass
     * @return
     */
    private static MinioClient initMinio(String minioUrl, String minioName,String minioPass) {
        if (minioClient == null) {
            try {
                minioClient = MinioClient.builder()
                        .endpoint(minioUrl)
                        .credentials(minioName, minioPass)
                        .build();
            } catch (Exception e) {
                e.printStackTrace();
            }
        }
        return minioClient;
    }

    /**
     * 上传文件到minio
     * @param stream
     * @param relativePath
     * @return
     */
    public static String upload(InputStream stream,String relativePath) throws Exception {
        initMinio(minioUrl, minioName,minioPass);
        if(minioClient.bucketExists(BucketExistsArgs.builder().bucket(bucketName).build())) {
            log.info("Bucket already exists.");
        } else {
            // 创建一个名为ota的存储桶
            minioClient.makeBucket(MakeBucketArgs.builder().bucket(bucketName).build());
            log.info("create a new bucket.");
        }
        PutObjectArgs objectArgs = PutObjectArgs.builder().object(relativePath)
                .bucket(bucketName)
                .contentType("application/octet-stream")
                .stream(stream,stream.available(),-1).build();
        minioClient.putObject(objectArgs);
        stream.close();
        return minioUrl+bucketName+"/"+relativePath;
    }

    /**
     * 列出桶内所有对象(added by caixibei)
     * @param bucketName 存储桶名称
     * @param prefix 对象名称的前缀,列出有该前缀的对象,如果为null ,表示查全部
     * @param recursive 是否递归查找,如果是false,就模拟文件夹结构查找
     */
    public static List<Map<String,String>> listObjects(String bucketName, String prefix, boolean recursive,boolean useVersion1) throws Exception {
        initMinio(minioUrl, minioName,minioPass);
        List<Map<String,String>> objLists = new ArrayList<>();
        // 检查桶是否存在
        boolean found = minioClient.bucketExists(BucketExistsArgs.builder().bucket(bucketName).build());
        if (found) {
            // 列出桶里的对象
            Iterable<Result<Item>> myObjects = minioClient.listObjects(ListObjectsArgs.builder().bucket(bucketName).prefix(prefix).recursive(recursive).useApiVersion1(useVersion1).build());
            for (Result<Item> result : myObjects) {
                Map<String,String> map = new HashMap<>();
                Item item = result.get();
                map.put("lastModified",item.lastModified().format(DateTimeFormatter.BASIC_ISO_DATE));
                map.put("size",String.valueOf(item.size()));
                map.put("fileName",item.objectName());
                map.put("url",getObjectUrl(bucketName,Method.GET,item.objectName()));
                objLists.add(map);
            }
        } else {
            throw new Exception("【Error】不存在桶:"+bucketName);
        }
        return objLists;
    }

    /**
     * 获取某一个存储对象的下载链接(added by caixibei)
     * @param bucketName 桶名
     * @param method  方法类型
     * @param objectName 对象名
     * @return url 下载链接
     * @throws ServerException 服务异常
     * @throws InsufficientDataException
     * @throws ErrorResponseException
     * @throws IOException
     * @throws NoSuchAlgorithmException
     * @throws InvalidKeyException
     * @throws InvalidResponseException
     * @throws XmlParserException
     * @throws InternalException
     */
    public static String getObjectUrl(String bucketName,Method method,String objectName) throws ServerException, InsufficientDataException, ErrorResponseException, IOException, NoSuchAlgorithmException, InvalidKeyException, InvalidResponseException, XmlParserException, InternalException {
        return minioClient.getPresignedObjectUrl(GetPresignedObjectUrlArgs.builder()
                .method(method)
                .bucket(bucketName)
                .object(objectName).build());
    }
}

4.案例

代码语言:java
AI代码解释
复制
/**
* minio测试接口--上传文件
* @param multipartFile
* @return
*/
@GetMapping("/index/minioTest")
@ApiOperation(value = "minio测试接口",notes = "minio测试接口")
public @ResponseBody String minioTest(@RequestParam("file") MultipartFile multipartFile){
    MinioUtils.upload(multipartFile,"","test");
    return "上传成功!";
}

/**
* minio测试接口-获取文件列表
* @return
* @throws Exception
*/
@GetMapping("/index/minioTest2")
@ApiOperation(value = "minio测试接口-获取文件列表",notes = "minio测试接口-获取文件列表")
public @ResponseBody JSONArray minioTest2() throws Exception {
    return JSONUtil.parseArray(MinioUtils.listObjects("test",null,true,false));
}

原创声明:本文系作者授权腾讯云开发者社区发表,未经许可,不得转载。

如有侵权,请联系 cloudcommunity@tencent.com 删除。

原创声明:本文系作者授权腾讯云开发者社区发表,未经许可,不得转载。

如有侵权,请联系 cloudcommunity@tencent.com 删除。

评论
登录后参与评论
暂无评论
推荐阅读
编辑精选文章
换一批
🗄️Spring Boot 3 整合 MinIO 实现分布式文件存储
文件存储已成为一个做任何应用都不可回避的需求。传统的单机文件存储方案在面对大规模数据和高并发访问时往往力不从心,而分布式文件存储系统则提供了更好的解决方案。本篇文章我将基于Spring Boot 3 为大家讲解如何基于MinIO来实现分布式文件存储。
别惹CC
2025/03/05
8411
🗄️Spring Boot 3 整合 MinIO 实现分布式文件存储
手把手教你SpringBoot轻松整合Minio实现文件上传下载
前面我们介绍了什么是分布式存储系统,介绍了什么是MinIO,最后如何使用MinIO构建分布式文件系统。那么怎么在实际的项目中使用MinIO呢?接下来就手把手教你如何在SpringBoot中轻松整合MinIO 。
章为忠学架构
2023/10/06
5.8K0
手把手教你SpringBoot轻松整合Minio实现文件上传下载
minio | docker
在下是首席架构师
2023/10/25
4811
minio | docker
springboot集成minio,docker部署
知识浅谈
2023/09/25
4070
SpringBoot整合Minio 项目中使用自己文件存储服务器
minio介绍: MinIO是根据GNU Affero通用公共许可证v3.0发布的高性能对象存储。 史上最详细Docker安装Minio
宁在春
2022/10/31
2.5K0
SpringBoot整合Minio 项目中使用自己文件存储服务器
SpringBoot × MinIO 极速开发指南:对象存储服务高可用实战
MinIO的在Linux上的部署可以参考:MinIO在Linux上的安装与部署_minio linux部署-CSDN博客
摘星.
2025/05/20
2840
SpringBoot × MinIO 极速开发指南:对象存储服务高可用实战
springboot整合minio
accessKey和secretKey需要根据启动的窗口上的值进行配置bucketName这个需要自己创建
java后端指南
2022/02/23
1.3K0
springboot整合minio
MinIO的使用(内含docker的简单使用)
MinIO 是全球领先的对象存储先锋,在标准硬件上,读/写速度上高达183 GB / 秒 和 171 GB / 秒。MinIO用作云原生应用程序的主要存储,与传统对象存储相比,云原生应用程序需要更高的吞吐量和更低的延迟。通过添加更多集群可以扩展名称空间,更多机架,直到实现目标。同时,符合一切原生云计算的架构和构建过程,并且包含最新的云计算的全新的技术和概念。
花花522
2023/03/07
4.6K0
MinIO的使用(内含docker的简单使用)
写给大忙人看的 – Java中上传文件MinIO服务器(2)
上一篇 写给大忙人看的 – 搭建文件服务器 MinIO(一),我们已经成功地搭建了 MinIO 文件服务器,这一篇讲解在 Java 中如何上传文件至 MinIO
全栈程序员站长
2022/09/05
1.1K0
接入minio我来帮你做
​ hello,everyone。好久不见,相信大家日常开发工作中对于文件的存储,读取等都是有大大小小的需求的。在博主刚接触springboot的时候,比较喜欢把一些文件存储在linux的磁盘上,但是后面发现对于磁盘上的文件管理很麻烦。而且如果磁盘一旦损坏,那么存储在磁盘上的文件将会全部丢失。为了解决上面的需求与问题,本文将给大家带来分布式文件存储中间件-minio。
柏炎
2022/08/23
1.5K0
接入minio我来帮你做
SpringBoot 整合 Minio
MinIO 是一个基于 Go 实现的高性能、兼容 S3 协议的对象存储。它采用 GNU AGPL v3 开源协议,项目地址是 https://github.com/minio/minio 。
默存
2023/09/01
6920
SpringBoot 整合 Minio
02-SpringBoot集成MinIo
建议不要用MinIo自带的图片分享URL, 因为最大只支持7天, 我在网上找了一些方案, 基本都是直接通过文件夹路径访问, 但是只有单击可以, 我在集群上发现是不行的, 所以我是自己做了个接口直接返回图片流
彼岸舞
2022/08/24
2.3K0
02-SpringBoot集成MinIo
SpringBoot + Minio 实现文件切片极速上传技术(实战典藏版)
提示:请不要多个博客来回跳着看,此文章之详细绝无仅有,融合多家之长,如果遇见报错,请仔细捋一遍文章,不要忽略!我在写的时候因为许多文章不全面,来回跳遇见许多坑,希望大家可以避免,本文章中悉数做了标注提醒!!!
码猿技术专栏
2024/07/08
2.3K0
SpringBoot + Minio 实现文件切片极速上传技术(实战典藏版)
springboot整合minio+vue实现大文件分片上传,断点续传(复制可用,包含minio工具类)
一只牛博
2025/05/31
6370
springboot整合minio+vue实现大文件分片上传,断点续传(复制可用,包含minio工具类)
Spring Boot 工程集成 minio 对象存储
上一片博文里我们介绍了 minio 对象存储的部署,这篇博文 我们主要介绍下 基于minio和spring boot的工程开发
Freedom123
2024/03/29
2030
在Java中使用MinIO:实现对象存储的便捷与高效
随着云计算和大数据技术的快速发展,对象存储已成为现代应用架构中不可或缺的一部分。MinIO是一个高性能、开源的对象存储服务器,兼容Amazon S3 API,非常适合用于存储大量非结构化数据。本文将详细介绍如何在Java中使用MinIO,帮助开发者快速上手并充分利用其强大的功能。
Front_Yue
2024/10/12
2.2K0
在Java中使用MinIO:实现对象存储的便捷与高效
MinIO服务部署指南
‌‌MinIO是一个基于‌Apache License v2.0开源协议的对象存储服务‌,它兼容‌亚马逊S3云存储服务接口,非常适合存储大容量非结构化的数据,如图片、视频、日志文件、备份数据和容器/虚拟机镜像等,文件大小可以从几KB到最大5TB不等
ma布
2024/10/30
5740
MinIO服务部署指南
Spring Boot整合MinIO实现文件上传和下载
家人们啦!,上篇文章了,我们讲了如何使用docker-compose快速部署MinIO,在今天的文章中,我将向大家介绍如何将Spring Boot与MinIO进行无缝整合,以便高效地管理和操作文件存储。通过这个整合,你将能够轻松地在Spring Boot应用程序中实现文件的上传和下载等功能。让我们开始吧
修己xj
2023/08/25
1.2K0
Spring Boot整合MinIO实现文件上传和下载
Spring Boot 3 整合 minio 实现文件存储
就拿小型网站系统来说,一般情况下,会把文件存储服务和网站系统部署在同一台服务器上。这种做法在小成本资源投入方面确实具有一定优势。如果网站的访问量处于较低水平,这种方式基本不会引发什么问题。然而,随着网站访问量的逐步攀升,网站文件资源的读取操作变得越来越频繁。此时,单台服务器可能就无法承受如此大量的请求了,进而导致网站打不开或者出现系统异常等情况。
Harry技术
2025/01/13
4460
Spring Boot 3 整合 minio 实现文件存储
SpringBoot 整合 Mini
这里把MinioClient注入到Bean中,后续使用MinioClient调用Minio的API。
阿珍
2024/10/28
2060
SpringBoot 整合 Mini
相关推荐
🗄️Spring Boot 3 整合 MinIO 实现分布式文件存储
更多 >
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档