SpringBoot使用Minio進行文件存儲的實現(xiàn)
一、minio
MinIO
是一個高性能的對象存儲原生支持 Kubernetes
部署的解決方案。 MinIO
提供了一個 Amazon Web Services S3
兼容 API
并支持所有核心 S3
功能。
MinIO
對象存儲使用 buckets
來組織對象。 存儲桶類似于文件系統(tǒng)中的文件夾或目錄,其中每個 桶可以容納任意數(shù)量的對象。 MinIO
存儲桶提供 與 AWS S3
存儲桶相同的功能。
其中 MinIO
的優(yōu)勢有:
高性能:
MinIO
是全球領(lǐng)先的對象存儲先鋒,在標(biāo)準(zhǔn)硬件上,讀/寫速度上高達183 GB / 秒
和 171 GB / 秒
。
可擴展性:
MinIO
利用了web
縮放器的來之不易的知識,為對象存儲帶來了簡單的存儲縮放模型, 在 MinIO
, 擴展從單個群集開始,該群集可以與其他MinIO
群集聯(lián)合以創(chuàng)建全局名稱空間, 并在需要時可以跨越多個不同的數(shù)據(jù)中心。 通過添加更多集群可以擴展名稱空間, 更多機架,直到實現(xiàn)目標(biāo)。
云原生支持:
MinIO
是在過去4年的時間內(nèi)從0開始打造的一款軟件 ,符合一切原生云計算的架構(gòu)和構(gòu)建過程,并且包含最新的云計算的全新的技術(shù)和概念。 其中包括支持Kubernetes
、微服和多租戶的的容器技術(shù)。使對象存儲對于 Kubernetes
更加友好。
源碼開放與企業(yè)級支持:
MinIO
基于Apache V2 license 100%
開放源代碼 。 這就意味著 MinIO
的客戶能夠自動的、無限制、自由免費使用和集成MinIO
、自由的創(chuàng)新和創(chuàng)造、 自由的去修改、自由的再次發(fā)行新的版本和軟件. 確實, MinIO
強有力的支持和驅(qū)動了很多世界500強的企業(yè)。 此外,其部署的多樣性和專業(yè)性提供了其他軟件無法比擬的優(yōu)勢。
在實驗開始前請確保安裝完成了 minio
:
二、SpringBoot 使用 Minio 進行文件存儲
首先新建一個 SpringBoot
項目,在 pom
中引入 minio
依賴:
<dependency> <groupId>io.minio</groupId> <artifactId>minio</artifactId> <version>8.2.1</version> </dependency>
在配置文件中,聲明出 minio
的信息:
minio: url: http://192.168.40.169:9000 # minio配置的地址,端口9000,注意不是控制臺的端口 accessKey: minioadmin # 賬號 secretKey: minioadmin # 密碼 bucketName: test-bucket # MinIO桶名字
下面創(chuàng)建一個配置類,對 MinioClient
進行創(chuàng)建:
@Data @Configuration @ConfigurationProperties(prefix = "minio") public class MinioConfig { /** * 服務(wù)地址 */ private String url; /** * 用戶名 */ private String accessKey; /** * 密碼 */ private String secretKey; /** * 存儲桶名稱 */ private String bucketName; @Bean public MinioClient getMinioClient() throws Exception { MinioClient minioClient = MinioClient.builder().endpoint(url).credentials(accessKey, secretKey).build(); //判斷桶是否存在,不存在則新建 if (!minioClient.bucketExists(BucketExistsArgs.builder().bucket(bucketName).build())){ minioClient.makeBucket(MakeBucketArgs.builder() .bucket(bucketName) .build()); } return minioClient; } }
下面創(chuàng)建一個工具類 MinioTool
將常用的操作封裝在工具類中:
@Component public class MinioTool { @Autowired private MinioClient minioClient; @Autowired private MinioConfig minioConfig; /** * 查看存儲bucket是否存在 * * @param bucketName 存儲bucket * @return boolean */ public Boolean bucketExists(String bucketName) { Boolean found; try { found = minioClient.bucketExists(BucketExistsArgs.builder().bucket(bucketName).build()); } catch (Exception e) { e.printStackTrace(); return false; } return found; } /** * 創(chuàng)建存儲bucket * * @param bucketName 存儲bucket名稱 * @return Boolean */ public Boolean makeBucket(String bucketName) { try { minioClient.makeBucket(MakeBucketArgs.builder() .bucket(bucketName) .build()); } catch (Exception e) { e.printStackTrace(); return false; } return true; } /** * 刪除存儲bucket * * @param bucketName 存儲bucket名稱 * @return Boolean */ public Boolean removeBucket(String bucketName) { try { minioClient.removeBucket(RemoveBucketArgs.builder() .bucket(bucketName) .build()); } catch (Exception e) { e.printStackTrace(); return false; } return true; } /** * 查看文件對象 * * @param bucketName 存儲bucket名稱 * @return 存儲bucket內(nèi)文件對象信息 */ public Iterable<Result<Item>> listObjects(String bucketName) { Iterable<Result<Item>> results = minioClient.listObjects( ListObjectsArgs.builder().bucket(bucketName).build()); return results; } /** * 批量刪除文件對象 * * @param bucketName 存儲bucket名稱 * @param objects 對象名稱集合 */ public Iterable<Result<DeleteError>> removeObjects(String bucketName, List<String> objects) { List<DeleteObject> dos = objects.stream().map(e -> new DeleteObject(e)).collect(Collectors.toList()); Iterable<Result<DeleteError>> results = minioClient.removeObjects(RemoveObjectsArgs.builder().bucket(bucketName).objects(dos).build()); return results; } /** * 文件上傳 * 文件名稱相同會覆蓋 * @param file 文件 * @return Boolean */ public Boolean upload(MultipartFile file, String fileName) { try { if (!bucketExists(minioConfig.getBucketName())) { makeBucket(minioConfig.getBucketName()); } PutObjectArgs objectArgs = PutObjectArgs.builder().bucket(minioConfig.getBucketName()).object(fileName) .stream(file.getInputStream(), file.getSize(), -1).contentType(file.getContentType()).build(); minioClient.putObject(objectArgs); } catch (Exception e) { e.printStackTrace(); return false; } return true; } /** * 文件下載 * * @param fileName 文件名稱 * @param res response * @return Boolean */ public void download(String fileName, HttpServletResponse res) { GetObjectArgs objectArgs = GetObjectArgs.builder().bucket(minioConfig.getBucketName()) .object(fileName).build(); try (GetObjectResponse response = minioClient.getObject(objectArgs)) { byte[] buf = new byte[1024]; int len; try (FastByteArrayOutputStream os = new FastByteArrayOutputStream()) { while ((len = response.read(buf)) != -1) { os.write(buf, 0, len); } os.flush(); byte[] bytes = os.toByteArray(); res.setCharacterEncoding("utf-8"); //設(shè)置強制下載不打開 res.setContentType("application/force-download"); res.addHeader("Content-Disposition", "attachment;fileName=" + fileName); try (ServletOutputStream stream = res.getOutputStream()) { stream.write(bytes); stream.flush(); } } } catch (Exception e) { e.printStackTrace(); } } public String getFileUrl(String fileName){ return StringFormatter.concat(minioConfig.getUrl(), "/", minioConfig.getBucketName(), "/", fileName).getValue(); } }
編寫測試接口,進行測試:
@Component public class MinioTool { @Autowired private MinioClient minioClient; @Autowired private MinioConfig minioConfig; /** * 查看存儲bucket是否存在 * * @param bucketName 存儲bucket * @return boolean */ public Boolean bucketExists(String bucketName) { Boolean found; try { found = minioClient.bucketExists(BucketExistsArgs.builder().bucket(bucketName).build()); } catch (Exception e) { e.printStackTrace(); return false; } return found; } /** * 創(chuàng)建存儲bucket * * @param bucketName 存儲bucket名稱 * @return Boolean */ public Boolean makeBucket(String bucketName) { try { minioClient.makeBucket(MakeBucketArgs.builder() .bucket(bucketName) .build()); } catch (Exception e) { e.printStackTrace(); return false; } return true; } /** * 刪除存儲bucket * * @param bucketName 存儲bucket名稱 * @return Boolean */ public Boolean removeBucket(String bucketName) { try { minioClient.removeBucket(RemoveBucketArgs.builder() .bucket(bucketName) .build()); } catch (Exception e) { e.printStackTrace(); return false; } return true; } /** * 查看文件對象 * * @param bucketName 存儲bucket名稱 * @return 存儲bucket內(nèi)文件對象信息 */ public Iterable<Result<Item>> listObjects(String bucketName) { Iterable<Result<Item>> results = minioClient.listObjects( ListObjectsArgs.builder().bucket(bucketName).build()); return results; } /** * 批量刪除文件對象 * * @param bucketName 存儲bucket名稱 * @param objects 對象名稱集合 */ public Iterable<Result<DeleteError>> removeObjects(String bucketName, List<String> objects) { List<DeleteObject> dos = objects.stream().map(e -> new DeleteObject(e)).collect(Collectors.toList()); Iterable<Result<DeleteError>> results = minioClient.removeObjects(RemoveObjectsArgs.builder().bucket(bucketName).objects(dos).build()); return results; } /** * 文件上傳 * 文件名稱相同會覆蓋 * @param file 文件 * @return Boolean */ public Boolean upload(MultipartFile file, String fileName) { try { if (!bucketExists(minioConfig.getBucketName())) { makeBucket(minioConfig.getBucketName()); } PutObjectArgs objectArgs = PutObjectArgs.builder().bucket(minioConfig.getBucketName()).object(fileName) .stream(file.getInputStream(), file.getSize(), -1).contentType(file.getContentType()).build(); minioClient.putObject(objectArgs); } catch (Exception e) { e.printStackTrace(); return false; } return true; } /** * 文件下載 * * @param fileName 文件名稱 * @param res response * @return Boolean */ public void download(String fileName, HttpServletResponse res) { GetObjectArgs objectArgs = GetObjectArgs.builder().bucket(minioConfig.getBucketName()) .object(fileName).build(); try (GetObjectResponse response = minioClient.getObject(objectArgs)) { byte[] buf = new byte[1024]; int len; try (FastByteArrayOutputStream os = new FastByteArrayOutputStream()) { while ((len = response.read(buf)) != -1) { os.write(buf, 0, len); } os.flush(); byte[] bytes = os.toByteArray(); res.setCharacterEncoding("utf-8"); //設(shè)置強制下載不打開 res.setContentType("application/force-download"); res.addHeader("Content-Disposition", "attachment;fileName=" + fileName); try (ServletOutputStream stream = res.getOutputStream()) { stream.write(bytes); stream.flush(); } } } catch (Exception e) { e.printStackTrace(); } } public String getFileUrl(String fileName){ return StringFormatter.concat(minioConfig.getUrl(), "/", minioConfig.getBucketName(), "/", fileName).getValue(); } }
三、測試
測試上傳文件:
如果使用 返回的 url 直接訪問文件,可以發(fā)現(xiàn)返回權(quán)限不足:
這里需要改一下 Bucket
的 Access Policy
,默認(rèn)為 private
,可以修改為 public
便無需認(rèn)證,但安全性無法保證:
再次進行訪問,文件就可以打開了:
如果需要保持 private
,可以通過 MinioClient
進行下載,使用 download
測試接口下載文件:http://localhost:8080/file/download/20cab4e3979eba6003f95aca0dc75c63.jpg
到此這篇關(guān)于SpringBoot使用Minio進行文件存儲的實現(xiàn)的文章就介紹到這了,更多相關(guān)SpringBoot Minio文件存儲內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
淺談在Java中使用Callable、Future進行并行編程
這篇文章主要介紹了淺談在Java中使用Callable、Future進行并行編程,具有一定借鑒價值,需要的朋友可以參考下。2017-12-12使用SpringBoot發(fā)送郵箱驗證碼的簡單實現(xiàn)
這篇文章主要介紹了使用SpringBoot發(fā)送郵箱驗證碼的簡單實現(xiàn),咱們今天來講使用QQ郵箱來發(fā)送和接收驗證碼,首先來介紹一下它在SpringBoot項目中的具體應(yīng)用,需要的朋友可以參考下2023-04-04JAVA8 List<List<Integer>> list中再裝一個list轉(zhuǎn)成一個list操
這篇文章主要介紹了JAVA8 List<List<Integer>> list中再裝一個list轉(zhuǎn)成一個list操作,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2020-08-08Java 使用IO流實現(xiàn)大文件的分割與合并實例詳解
這篇文章主要介紹了Java 使用IO流實現(xiàn)大文件的分割與合并實例詳解的相關(guān)資料,需要的朋友可以參考下2016-12-12IntelliJ IDEAx導(dǎo)出安卓(Android)apk文件圖文教程
這篇文章主要為大家詳細(xì)介紹了IntelliJ IDEAx導(dǎo)出安卓(Android)apk文件圖文教程,文中步驟介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下2020-10-10MyBatis學(xué)習(xí)教程(七)-Mybatis緩存介紹
MyBatis緩存分為一級緩存和二級緩存一級緩存,本文給大家介紹mybatis緩存知識,非常不錯具有參考借鑒價值,感興趣的朋友一起學(xué)習(xí)吧2016-05-05Spring框架設(shè)值注入操作實戰(zhàn)案例分析
這篇文章主要介紹了Spring框架設(shè)值注入操作,結(jié)合具體實例形式分析了spring框架設(shè)值注入相關(guān)實現(xiàn)與使用方法,需要的朋友可以參考下2019-11-11