Springboot集成阿里云OSS上传

API 接口描述

DEMO提供的四个API接口,支持不同方式的文件和 JSON 数据上传:

1. 普通文件上传接口

上传任意类型的文件

2. JSON 字符串上传接口

上传 JSON 字符串

3. 单个 JSON 压缩上传接口

上传并压缩 JSON 字符串

4. 批量 JSON 压缩上传接口

批量上传并压缩多个 JSON 字符串

AliOssConfig

/*** @author Melody* @description:* @date 2025-05-06*/
@Data
@Configuration
public class AliOssConfig {@Value("${aliyun.oss.endpoint}")private String endpoint;@Value("${aliyun.oss.access-key-id}")private String accessKeyId;@Value("${aliyun.oss.access-key-secret}")private String accessKeySecret;@Value("${aliyun.oss.bucket-name}")private String bucketName;
}

AliOssUtil

/*** @author Melody* @description:* @date 2025-05-06*/
@Component
@Slf4j
public class AliOssUtil {private final AliOssConfig aliOssConfig;private OSS ossClient;@Autowiredpublic AliOssUtil(AliOssConfig aliOssConfig) {this.aliOssConfig = aliOssConfig;ossClient = new OSSClientBuilder().build(aliOssConfig.getEndpoint(), aliOssConfig.getAccessKeyId(), aliOssConfig.getAccessKeySecret());}public String uploadFile(MultipartFile file, String fileType) {if (file.isEmpty()) {throw new IllegalArgumentException("文件不能为空");}try {// 生成文件名和路径String fileName = generateUniqueFileName(file.getOriginalFilename());String filePath = generateDateBasedFilePath(fileType, fileName);// 上传文件try (InputStream inputStream = file.getInputStream()) {ossClient.putObject(aliOssConfig.getBucketName(), filePath, inputStream);}// 返回 URLreturn buildOssUrl(filePath);} catch (IOException e) {throw new RuntimeException("文件读取失败: " + e.getMessage());} catch (OSSException | ClientException e) {throw new RuntimeException("OSS服务异常: " + e.getMessage());}}/*** 生成唯一文件名(UUID + 扩展名)*/private String generateUniqueFileName(String extension) {return UUID.randomUUID() + extension;}/*** 提取文件扩展名(处理无扩展名的情况)*/private String getFileExtension(String originalFilename) {if (originalFilename == null) return ".dat";int lastDotIndex = originalFilename.lastIndexOf(".");return (lastDotIndex == -1) ? ".dat" : originalFilename.substring(lastDotIndex);}/*** 生成基于日期的存储路径(如 images/2024/06/15/uuid.jpg)*/private String generateDateBasedFilePath(String fileType, String fileName) {if (StringUtils.isEmpty(fileType)) {fileType = "file";}String datePath = LocalDate.now().format(DateTimeFormatter.ofPattern("yyyy/MM/dd"));return String.format(fileType + "/%s/%s", datePath, fileName);}/*** 构建 OSS 访问 URL*/private String buildOssUrl(String filePath) {return String.format("https://%s.%s/%s",aliOssConfig.getBucketName(),aliOssConfig.getEndpoint(),filePath);}/*** 上传json文件到OSS* @param json* @param fileType* @param fileName*/public String uploadJSONToOSS(String json, String fileType, String fileName) {// 创建 OSSClient 实例try {String filePath = generateDateBasedFilePath(fileType, fileName + ".json");// 将 JSON 字符串转换为字节数组输入流ByteArrayInputStream inputStream = new ByteArrayInputStream(json.getBytes(StandardCharsets.UTF_8));PutObjectRequest putObjectRequest = new PutObjectRequest(aliOssConfig.getBucketName(), filePath, inputStream);ossClient.putObject(putObjectRequest);return buildOssUrl(filePath);} catch (Exception e) {log.error("文件上传失败:" + e.getMessage());return "";}}/*** 将JSON内容压缩为ZIP并上传到OSS* @param json JSON字符串* @param fileType 文件分类(如images、docs)* @param fileName 文件名(不含扩展名)* @return 上传后的OSS访问URL*/public String uploadCompressedJSONToOSS(String json, String fileType, String fileName) {ByteArrayOutputStream baos = new ByteArrayOutputStream();try (ZipOutputStream zos = new ZipOutputStream(baos)) {// 创建ZIP条目并添加JSON内容ZipEntry zipEntry = new ZipEntry(fileName + ".json");zos.putNextEntry(zipEntry);zos.write(json.getBytes(StandardCharsets.UTF_8));zos.closeEntry();// 确保ZIP流完全关闭后再获取字节数组} catch (IOException e) {log.error("JSON压缩失败", e);return "";}// 生成带日期的ZIP文件路径String filePath = generateDateBasedFilePath(fileType, fileName + ".zip");// 上传到OSStry (ByteArrayInputStream bais = new ByteArrayInputStream(baos.toByteArray())) {OSS ossClient = new OSSClientBuilder().build(aliOssConfig.getEndpoint(),aliOssConfig.getAccessKeyId(),aliOssConfig.getAccessKeySecret());try {PutObjectRequest putRequest = new PutObjectRequest(aliOssConfig.getBucketName(),filePath,bais);ossClient.putObject(putRequest);return buildOssUrl(filePath);} finally {ossClient.shutdown();}} catch (IOException e) {log.error("ZIP文件上传失败", e);return "";}}/*** 将多个JSON内容压缩为ZIP并上传到OSS* @param jsonMap JSON字符串映射,键为文件名(不含扩展名),值为JSON内容* @param fileType 文件分类(如images、docs)* @param zipFileName 生成的ZIP文件名(不含扩展名)* @return 上传后的OSS访问URL*/public String uploadMultipleJSONsToOSS(Map<String, String> jsonMap, String fileType, String zipFileName) {ByteArrayOutputStream baos = new ByteArrayOutputStream();try (ZipOutputStream zos = new ZipOutputStream(baos)) {// 遍历所有JSON内容,为每个JSON创建ZIP条目for (Map.Entry<String, String> entry : jsonMap.entrySet()) {String fileName = entry.getKey();String jsonContent = entry.getValue();// 创建ZIP条目并添加JSON内容ZipEntry zipEntry = new ZipEntry(fileName + ".json");zos.putNextEntry(zipEntry);zos.write(jsonContent.getBytes(StandardCharsets.UTF_8));zos.closeEntry();}} catch (IOException e) {log.error("JSON压缩失败", e);return "";}// 生成带日期的ZIP文件路径String filePath = generateDateBasedFilePath(fileType, zipFileName + ".zip");// 上传到OSStry (ByteArrayInputStream bais = new ByteArrayInputStream(baos.toByteArray())) {OSS ossClient = new OSSClientBuilder().build(aliOssConfig.getEndpoint(),aliOssConfig.getAccessKeyId(),aliOssConfig.getAccessKeySecret());try {PutObjectRequest putRequest = new PutObjectRequest(aliOssConfig.getBucketName(),filePath,bais);ossClient.putObject(putRequest);return buildOssUrl(filePath);} finally {ossClient.shutdown();}} catch (IOException e) {log.error("ZIP文件上传失败", e);return "";}}/*** 删除文件*/public void deleteFile(String filePath) {ossClient.deleteObject(aliOssConfig.getBucketName(), filePath);}
}    

AliOssController

/*** @author Melody* @description:* @date 2025-05-06*/
@RestController
@RequestMapping("/system/file")
public class AliOssController {private final AliOssUtil aliOssUtil;public AliOssController(AliOssUtil aliOssUtil) {this.aliOssUtil = aliOssUtil;}/*** 阿里云OSS文件上传*/@PostMapping("/upload")public SysFileDTO uploadFile(MultipartFile file,@RequestParam(value = "fileType", required = false) String fileType) {String url = aliOssUtil.uploadFile(file, fileType);logger.info("上传文件路径:" + url);return SysFileDTO.builder().url(url).build();}/*** 阿里云OSS文件上传*/@PostMapping("/uploadJSONToOSS")public SysFileDTO uploadJSONToOSS(@RequestParam(value = "jsonData") String jsonData) {String url = aliOssUtil.uploadJSONToOSS(jsonData, "json", "example");logger.info("上传文件路径:" + url);return SysFileDTO.builder().url(url).build();}/*** 阿里云OSS文件上传压缩后的json文件*/@PostMapping("/uploadCompressedJSONToOSS")public SysFileDTO uploadCompressedJSONToOSS(@RequestParam(value = "jsonData") String jsonData) {String url = aliOssUtil.uploadCompressedJSONToOSS(jsonData, "json", "example");logger.info("上传文件路径:" + url);return SysFileDTO.builder().url(url).build();}/*** 阿里云OSS文件上传压缩后的json文件*/@PostMapping("/uploadMultipleJSONsToOSS")public SysFileDTO uploadMultipleJSONsToOSS(@RequestParam(value = "jsonData") String jsonData) {Map<String, String> jsonMap = new HashMap<>();jsonMap.put("data1", jsonData);jsonMap.put("data2", "{\"key2\":\"value2\"}");String url = aliOssUtil.uploadMultipleJSONsToOSS(jsonMap, "json", "combined");logger.info("上传文件路径:" + url);return SysFileDTO.builder().url(url).build();}
}

application.yml

# 阿里云OSS配置
aliyun:oss:endpoint: endpointaccessKeyId: accessKeyIdaccessKeySecret: accessKeySecretbucketName: bucketName

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如若转载,请注明出处:http://www.pswp.cn/diannao/88051.shtml
繁体地址,请注明出处:http://hk.pswp.cn/diannao/88051.shtml
英文地址,请注明出处:http://en.pswp.cn/diannao/88051.shtml

如若内容造成侵权/违法违规/事实不符,请联系英文站点网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

删除大表数据注意事项

数据库是否会因删除操作卡死&#xff0c;没有固定的 “安全删除条数”&#xff0c;而是受数据库配置、表结构、操作方式、当前负载等多种因素影响。以下是关键影响因素及实践建议&#xff1a; 一、导致数据库卡死的核心因素 硬件与数据库配置 CPU / 内存瓶颈&#xff1a;删除…

Redis 是单线程模型?|得物技术

一、背景 使用过Redis的同学肯定都了解过一个说法&#xff0c;说Redis是单线程模型&#xff0c;那么实际情况是怎样的呢&#xff1f; 其实&#xff0c;我们常说Redis是单线程模型&#xff0c;是指Redis采用单线程的事件驱动模型&#xff0c;只有并且只会在一个主线程中执行Re…

[特殊字符] AIGC工具深度实战:GPT与通义灵码如何彻底重构企业开发流程

&#x1f50d; 第一模块&#xff1a;理念颠覆——为什么AIGC不是“玩具”而是“效能倍增器”&#xff1f; ▍企业开发的核心痛点图谱&#xff08;2025版&#xff09; ​​研发效能瓶颈​​&#xff1a;需求膨胀与交付时限矛盾持续尖锐&#xff0c;传统敏捷方法论已触天花板​…

(LeetCode 面试经典 150 题) 169. 多数元素(哈希表 || 二分查找)

题目&#xff1a;169. 多数元素 方法一&#xff1a;二分法&#xff0c;最坏的时间复杂度0(nlogn)&#xff0c;但平均0(n)即可。空间复杂度为0(1)。 C版本&#xff1a; int nnums.size();int l0,rn-1;while(l<r){int mid(lr)/2;int ans0;for(auto x:nums){if(xnums[mid]) a…

(17)java+ selenium->自动化测试-元素定位大法之By css上

1.简介 CSS定位方式和xpath定位方式基本相同,只是CSS定位表达式有其自己的格式。CSS定位方式拥有比xpath定位速度快,且比CSS稳定的特性。下面详细介绍CSS定位方式的使用方法。相对CSS来说,具有语法简单,定位速度快等优点。 2.CSS定位优势 CSS定位是平常使用过程中非常重要…

【软考高级系统架构论文】企业集成平台的技术与应用

论文真题 企业集成平台是一个支持复杂信息环境下信息系统开发、集成和协同运行的软件支撑环境。它基于各种企业经营业务的信息特征,在异构分布环境(操作系统、网络、数据库)下为应用提供一致的信息访问和交互手段,对其上运行的应用进行管理,为应用提供服务,并支持企业信息…

i.MX8MP LVDS 显示子系统全解析:设备树配置与 DRM 架构详解

&#x1f525; 推荐&#xff1a;《Yocto项目实战教程&#xff1a;高效定制嵌入式Linux系统》 京东正版促销&#xff0c;欢迎支持原创&#xff01; 链接&#xff1a;https://item.jd.com/15020438.html i.MX8MP LVDS 显示子系统全解析&#xff1a;设备树配置与 DRM 架构详解 在…

keep-alive实现原理及Vue2/Vue3对比分析

一、keep-alive基本概念 keep-alive是Vue的内置组件&#xff0c;用于缓存组件实例&#xff0c;避免重复渲染。它具有以下特点&#xff1a; 抽象组件&#xff1a;自身不会渲染DOM&#xff0c;也不会出现在父组件链中包裹动态组件&#xff1a;缓存不活动的组件实例&#xff0c;…

安卓jetpack compose学习笔记-Navigation基础学习

目录 一、Navigation 二、BottomNavigation Compose是一个偏向静态刷新的UI组件&#xff0c;如果不想要自己管理页面切换的复杂状态&#xff0c;可以以使用Navigation组件。 页面间的切换可以NavHost&#xff0c;使用底部页面切换栏&#xff0c;可以使用脚手架的bottomBarNav…

基于大数据技术的在UGC数据分析与路线推荐的研究

博主介绍&#xff1a;java高级开发&#xff0c;从事互联网行业六年&#xff0c;熟悉各种主流语言&#xff0c;精通java、python、php、爬虫、web开发&#xff0c;已经做了六年的毕业设计程序开发&#xff0c;开发过上千套毕业设计程序&#xff0c;没有什么华丽的语言&#xff0…

flask通过表单自动产生get请求的参数、form表单实现POST请求的自动提交

通过表单自动产生get请求的参数 相关代码如下&#xff1a; import flaskapp flask.Flask(__name__)app.route(/) def login():html <!DOCTYPE html><html lang"en"><head><meta charset"UTF-8"><title>flask表单实现get…

《情感反诈模拟器》2025学习版

1.2 专业内容支持 67篇情感诈骗案例研究14万字心理学分析资料783条专业配音对白 二、安装与运行 2.1 系统要求 最低配置&#xff1a; 显卡&#xff1a;GTX 1060CPU&#xff1a;i5-8400存储&#xff1a;25GB空间 2.2 运行步骤 解压游戏文件&#xff08;21.7GB&#xff09;…

预训练 vs. 微调:大模型落地的核心两步,究竟有何不同?

在人工智能领域&#xff0c;尤其是自然语言处理&#xff08;NLP&#xff09;和计算机视觉&#xff08;CV&#xff09;&#xff0c;大型模型如GPT系列、BERT、Stable Diffusion等取得了令人瞩目的成就。支撑这些模型广泛应用的关键技术流程&#xff0c;通常包含两个核心阶段&…

微信原生小程序转uniapp过程及错误总结

https://ask.dcloud.net.cn/article/35786 此文章尤为重要&#xff0c;可以使用辅助工具 1、this.setData 源代码&#xff1a; this.setData({dateTime: obj.dateTime, });需更换为 this.dateTime obj.dateTime2、cookie问题 在此文章有解释 https://blog.csdn.net/ni155…

关于Spring JBDC

一、什么是Spring JDBC&#xff1f; 什么是JDBC&#xff1f; JDBC&#xff08;Java Database Connectivity&#xff09;是 Java 语言访问数据库的标准 API&#xff0c;它定义了一组接口和类&#xff0c;允许 Java 程序与各种数据库进行交互。JDBC 提供了执行 SQL 语句、处理结果…

【SpringBoot】Spring Boot实现SSE实时推送实战

以下是一个完整的基于 Spring Boot 的 Server-Sent Events (SSE) 示例&#xff0c;包括服务端和客户端的实现。 一、服务端实现 1. 创建 Spring Boot 项目 首先&#xff0c;创建一个基本的 Spring Boot 项目&#xff0c;并添加 spring-boot-starter-web 依赖。在 pom.xml 中…

若依导出模板时设置动态excel下拉框(表连接的)

若依导出模板时设置动态excel下拉框&#xff08;表连接的&#xff09; 一、问题二、解决1、实体类2.1、临时使用2.2、统一工具类3、调用 一、问题 若依导出只能&#xff1b;使用dictType、combo、comboReadDict、readConverterExp这些来控制字典的导出下拉&#xff0c;如果不是…

Rabbitmq集成springboot 使用死信队列

一、何为死信队列 RabbitMQ的死信队列&#xff08;Dead Letter Queue&#xff0c;DLQ&#xff09;是一种特殊的队列机制&#xff0c;用于处理那些无法被正常消费的消息。这些消息可能由于各种原因无法被消费者正确处理&#xff0c;如果不加以处理&#xff0c;可能会导致队列堵塞…

Spring Boot 项目中 resources 文件读取

开发必备&#xff01;Spring Boot 项目中 resources 文件读取的 9 大方案详解 在 Spring Boot 项目中&#xff0c;resources 目录承载着大量的关键资源&#xff0c;如配置文件、模板文件、脚本资源、数据文件等。而如何以合适的方式高效、安全地读取这些资源&#xff0c;往往是…

力扣-1143.最长公共子序列

题目描述 给定两个字符串 text1 和 text2&#xff0c;返回这两个字符串的最长 公共子序列 的长度。如果不存在 公共子序列 &#xff0c;返回 0 。 一个字符串的 子序列 是指这样一个新的字符串&#xff1a;它是由原字符串在不改变字符的相对顺序的情况下删除某些字符&#xf…