springboot(3.2.5)初步集成MinIO(8.5.9)开发记录

2024-05-08 09:36

本文主要是介绍springboot(3.2.5)初步集成MinIO(8.5.9)开发记录,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

springboot初步集成MinIO开发记录

  • 说明
  • 一:引入maven依赖
  • 二:手动注入minioClient
  • 三:创建service类
  • 四:测试打印连接信息
  • 五:时区转化工具类
  • 六:常用操作演示

说明

  • 这里只是作者开发的记录,已备将来完善和复习!内容很简单,大佬请路过。

一:引入maven依赖

  • 在项目pom中添加minio maven依赖
<dependency><groupId>io.minio</groupId><artifactId>minio</artifactId><version>8.5.9</version>
</dependency>

二:手动注入minioClient

  • 由于minio官方没有提供spring-boot-starter-minio的启动类,所以,这里选择手动注入!
  • 在项目根目录下创建config软件包,创建config
    在这里插入图片描述
import io.minio.MinioClient;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;@Configuration
public class Config {//链式编程 构建MinioClient对象@Beanpublic MinioClient minioClient(){return MinioClient.builder()// .region("zh-east-8") 可选注意和minio的区域配置一致.endpoint("http://ip:9000").credentials("minioadmin","minioadmin") // 默认 用户和秘密.build();}
}

三:创建service类

在这里插入图片描述

import io.minio.MinioClient;
import jakarta.annotation.Resource;
import org.springframework.stereotype.Service;@Service
public class MinioService {@Resourceprivate MinioClient minioClient;public void testMinioClient(){System.out.print(minioClient);}
}

四:测试打印连接信息

  • 在测试类中测试打印连接信息
@SpringBootTest
class MinioApplicationTests {@Resourceprivate MinioService minioService;@Resourceprivate MinioClient minioClient;@Testvoid test() {minioService.testMinioClient();}
}	
  • 测试结果
io.minio.MinioClient@7302ff13

五:时区转化工具类

  • 如果你困惑于文件最后修改时间和代码程序查询结果不一致,参看Minio(官方docker版)容器部署时区问题研究记录
    在这里插入图片描述
package com.yang.miniostudy01.utils;import java.lang.annotation.Retention;
import java.time.Instant;
import java.time.ZoneId;
import java.time.ZonedDateTime;
import java.time.format.DateTimeFormatter;
import java.time.format.DateTimeFormatterBuilder;public class TimeConverter {// 使用自定义格式化模式private static final DateTimeFormatter isoFormatter = new DateTimeFormatterBuilder().append(DateTimeFormatter.ISO_LOCAL_DATE).appendLiteral('T').append(DateTimeFormatter.ISO_LOCAL_TIME).optionalStart().appendOffsetId().optionalEnd().toFormatter();private static final DateTimeFormatter customFormatter = new DateTimeFormatterBuilder().append(DateTimeFormatter.ISO_LOCAL_DATE).appendLiteral('T').append(DateTimeFormatter.ISO_LOCAL_TIME).appendLiteral('Z') // 添加字面值 'Z' 表示时区偏移量.toFormatter();/*** 将UTC时间转换为指定时区的时间,格式保持ISO-8601 默认时区为东八区* @param utcTimeString UTC时间字符串(ISO-8601格式)* @return 指定时区的时间字符串(ISO-8601格式)*/public static String convertUtcToLocal(String utcTimeString) {String zoneIdString="Asia/Shanghai";return convertUtcToLocal(utcTimeString,zoneIdString);}/*** 将UTC时间转换为指定时区的时间,格式保持ISO-8601* @param utcTimeString UTC时间字符串(ISO-8601格式)* @param zoneIdString 时区ID,如"Asia/Shanghai"* @return 指定时区的时间字符串(ISO-8601格式)*/public static String convertUtcToLocal(String utcTimeString, String zoneIdString) {Instant utcTime = Instant.parse(utcTimeString);ZonedDateTime localTime = utcTime.atZone(ZoneId.of(zoneIdString));DateTimeFormatter formatter = customFormatter.withZone(ZoneId.of(zoneIdString));return formatter.format(localTime);}/*** 将本地时间转换为UTC时间* @param localTimeString 本地时间字符串(ISO-8601格式)* @param zoneIdString 时区ID,如"Asia/Shanghai"* @return UTC时间字符串(ISO-8601格式)*/public static String convertLocalToUtc(String localTimeString) {String zoneIdString="Asia/Shanghai";return convertLocalToUtc(localTimeString,zoneIdString);}/*** 将本地时间转换为UTC时间* @param localTimeString 本地时间字符串(ISO-8601格式)* @param zoneIdString 时区ID,如"Asia/Shanghai"* @return UTC时间字符串(ISO-8601格式)*/public static String convertLocalToUtc(String localTimeString, String zoneIdString) {ZonedDateTime localTime = ZonedDateTime.parse(localTimeString, customFormatter.withZone(ZoneId.of(zoneIdString)));Instant utcTime = localTime.toInstant();return isoFormatter.format(utcTime.atZone(ZoneId.of("UTC")));}
}

六:常用操作演示

package com.yang.miniostudy01;import com.yang.miniostudy01.service.MinioService;
import com.yang.miniostudy01.utils.TimeConverter;
import io.minio.*;
import io.minio.errors.*;
import io.minio.http.Method;
import io.minio.messages.Bucket;
import io.minio.messages.Item;
import jakarta.annotation.Resource;
import org.junit.jupiter.api.Test;
import org.springframework.boot.test.context.SpringBootTest;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;
import java.lang.reflect.Field;
import java.security.InvalidKeyException;
import java.security.NoSuchAlgorithmException;
import java.time.ZonedDateTime;
import java.util.List;
import java.util.concurrent.TimeUnit;@SpringBootTest
class MinioApplicationTests {@Resourceprivate MinioService minioService;@Resourceprivate MinioClient minioClient;/*** 判断桶是否存在*/@Testvoid testBucketExists(){try {boolean exists = minioClient.bucketExists(BucketExistsArgs.builder().bucket("test1").build());System.out.println("桶存在:"+exists);} catch (ErrorResponseException | XmlParserException | InsufficientDataException | InternalException |ServerException | InvalidKeyException | InvalidResponseException | IOException |NoSuchAlgorithmException e) {throw new RuntimeException(e);}}/*** 创建桶*/@Testvoid testBucketCreate() throws Exception {boolean exists = minioClient.bucketExists(BucketExistsArgs.builder().bucket("test02").build());if (!exists){minioClient.makeBucket(MakeBucketArgs.builder().bucket("test02").build());}System.out.println("bucket已存在,无需创建");}/*** 获取桶列表*/@Testvoid testBucketList() throws Exception {List<Bucket> buckets = minioClient.listBuckets();buckets.forEach(bucket-> System.out.println(bucket.name()+"--"+bucket.creationDate()));}/*** 删除桶*/@Testvoid testBucketRemove() throws Exception {minioClient.removeBucket(RemoveBucketArgs.builder().bucket("test02").build());}/*** 上传文件*/@Testvoid testObjectPut() throws Exception{File file = new File("src/main/resources/picture/image.jpg");ObjectWriteResponse response = minioClient.putObject(PutObjectArgs.builder().bucket("test02").object("image.jpg").stream(new FileInputStream(file),file.length(),-1).build());System.out.print(response);}/*** 简单上传文件*/@Testvoid testObjectUpload() throws Exception {ObjectWriteResponse response = minioClient.uploadObject(UploadObjectArgs.builder().bucket("test02").object("image3.jpg").filename("src/main/resources/picture/image.jpg").build());System.out.println(response);}/*** 判断文件是否存在*/@Testvoid testObjectExists() throws Exception {StatObjectResponse response = minioClient.statObject(StatObjectArgs.builder().bucket("test02").object("image.jpg").build());System.out.println("修改前文件信息:"+response);String localLastModifiedTime = TimeConverter.convertUtcToLocal(response.lastModified().toString());ZonedDateTime localLastModified = ZonedDateTime.parse(localLastModifiedTime);//通过反射 修改文件信息 不影响minio存储的文件信息try {Field lastModifiedField = response.getClass().getDeclaredField("lastModified");lastModifiedField.setAccessible(true);lastModifiedField.set(response, localLastModified);} catch (NoSuchFieldException | IllegalAccessException e) {e.printStackTrace();}System.out.println("修改后文件信息:"+response);}/*** 获取文件访问路径*/@Testvoid testObjectUrl() throws Exception {String objectUrl = minioClient.getPresignedObjectUrl(GetPresignedObjectUrlArgs.builder().bucket("test02").object("image.jpg").method(Method.GET).expiry(5, TimeUnit.MINUTES).build());System.out.println(objectUrl);}/*** 配置桶自定义权限*/@Testvoid testBucketCreateWithPrivilege() throws Exception {String bucketName="test02";boolean exists = minioClient.bucketExists(BucketExistsArgs.builder().bucket(bucketName).build());if (!exists){minioClient.makeBucket(MakeBucketArgs.builder().bucket(bucketName).build());}else{System.out.println("bucket已存在,无需创建");}// 定义存储桶的访问策略为公开// 定义公共访问策略String policyJson = "{\n" +"    \"Version\": \"2012-10-17\",\n" +"    \"Statement\": [\n" +"        {\n" +"            \"Sid\": \"PublicRead\",\n" +"            \"Effect\": \"Allow\",\n" +"            \"Principal\": \"*\",\n" +"            \"Action\": \"s3:GetObject\",\n" +"            \"Resource\": \"arn:aws:s3:::" + bucketName + "/*\"\n" +"        }\n" +"    ]\n" +"}";minioClient.setBucketPolicy(SetBucketPolicyArgs.builder().bucket(bucketName).config(policyJson).build());}/*** 文件下载*/@Testpublic void testObjectDownload() throws Exception {GetObjectResponse objectFile = minioClient.getObject(GetObjectArgs.builder().bucket("test02").object("image.jpg").build());System.out.println(objectFile.transferTo(new FileOutputStream("src/main/resources/picture/test.jpg")));}/*** 获取桶内文件信息*/@Testpublic void testObjectList(){Iterable<Result<Item>> results = minioClient.listObjects(ListObjectsArgs.builder().bucket("test02").build());results.forEach((itemResult)->{try {Item item = itemResult.get();System.out.println(item.objectName()+" \t"+item.lastModified()+"\t"+item.size());} catch (ErrorResponseException | XmlParserException | ServerException | NoSuchAlgorithmException |InvalidResponseException | InsufficientDataException | InvalidKeyException | IOException |InternalException e) {throw new RuntimeException(e);}});}/*** 删除文件*/@Testpublic void testRemoveObject() throws Exception {minioClient.removeObject(RemoveObjectArgs.builder().bucket("test02").object("5g3pl8.jpg").build());}
}

这篇关于springboot(3.2.5)初步集成MinIO(8.5.9)开发记录的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/969969

相关文章

Spring事务传播机制最佳实践

《Spring事务传播机制最佳实践》Spring的事务传播机制为我们提供了优雅的解决方案,本文将带您深入理解这一机制,掌握不同场景下的最佳实践,感兴趣的朋友一起看看吧... 目录1. 什么是事务传播行为2. Spring支持的七种事务传播行为2.1 REQUIRED(默认)2.2 SUPPORTS2

怎样通过分析GC日志来定位Java进程的内存问题

《怎样通过分析GC日志来定位Java进程的内存问题》:本文主要介绍怎样通过分析GC日志来定位Java进程的内存问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、GC 日志基础配置1. 启用详细 GC 日志2. 不同收集器的日志格式二、关键指标与分析维度1.

Java进程异常故障定位及排查过程

《Java进程异常故障定位及排查过程》:本文主要介绍Java进程异常故障定位及排查过程,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、故障发现与初步判断1. 监控系统告警2. 日志初步分析二、核心排查工具与步骤1. 进程状态检查2. CPU 飙升问题3. 内存

java中新生代和老生代的关系说明

《java中新生代和老生代的关系说明》:本文主要介绍java中新生代和老生代的关系说明,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、内存区域划分新生代老年代二、对象生命周期与晋升流程三、新生代与老年代的协作机制1. 跨代引用处理2. 动态年龄判定3. 空间分

Java设计模式---迭代器模式(Iterator)解读

《Java设计模式---迭代器模式(Iterator)解读》:本文主要介绍Java设计模式---迭代器模式(Iterator),具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,... 目录1、迭代器(Iterator)1.1、结构1.2、常用方法1.3、本质1、解耦集合与遍历逻辑2、统一

Java内存分配与JVM参数详解(推荐)

《Java内存分配与JVM参数详解(推荐)》本文详解JVM内存结构与参数调整,涵盖堆分代、元空间、GC选择及优化策略,帮助开发者提升性能、避免内存泄漏,本文给大家介绍Java内存分配与JVM参数详解,... 目录引言JVM内存结构JVM参数概述堆内存分配年轻代与老年代调整堆内存大小调整年轻代与老年代比例元空

深度解析Java DTO(最新推荐)

《深度解析JavaDTO(最新推荐)》DTO(DataTransferObject)是一种用于在不同层(如Controller层、Service层)之间传输数据的对象设计模式,其核心目的是封装数据,... 目录一、什么是DTO?DTO的核心特点:二、为什么需要DTO?(对比Entity)三、实际应用场景解析

Java 线程安全与 volatile与单例模式问题及解决方案

《Java线程安全与volatile与单例模式问题及解决方案》文章主要讲解线程安全问题的五个成因(调度随机、变量修改、非原子操作、内存可见性、指令重排序)及解决方案,强调使用volatile关键字... 目录什么是线程安全线程安全问题的产生与解决方案线程的调度是随机的多个线程对同一个变量进行修改线程的修改操

从原理到实战深入理解Java 断言assert

《从原理到实战深入理解Java断言assert》本文深入解析Java断言机制,涵盖语法、工作原理、启用方式及与异常的区别,推荐用于开发阶段的条件检查与状态验证,并强调生产环境应使用参数验证工具类替代... 目录深入理解 Java 断言(assert):从原理到实战引言:为什么需要断言?一、断言基础1.1 语

深度解析Java项目中包和包之间的联系

《深度解析Java项目中包和包之间的联系》文章浏览阅读850次,点赞13次,收藏8次。本文详细介绍了Java分层架构中的几个关键包:DTO、Controller、Service和Mapper。_jav... 目录前言一、各大包1.DTO1.1、DTO的核心用途1.2. DTO与实体类(Entity)的区别1