当前位置: 首页 > news >正文

文件下载时利用redis的队列模式顺序下载文件,防止多文件任务下载导致OOM

1、controller层控制


@Resourceprivate RedissonClient redissonClient;@Slf4j
@Service
public class CustomerSettlementExportServiceImpl implements ICustomerSettlementExportService {
/*** 文件加入队列顺序导出** @param pubFileExportList 参数* @return 结果*/public AjaxResult pubFileExport(List<PubFileExport> pubFileExportList) {if (CollectionUtils.isEmpty(pubFileExportList)) {return AjaxResult.error("客户信息不能为空!");}RQueue<String> queue = redissonClient.getQueue("downloadQueue:file:export");for (PubFileExport pubFileExport : pubFileExportList) {try {queue.add(JSONObject.toJSONString(pubFileExport));} catch (Exception e) {log.error("加入队列失败=", e);return AjaxResult.error("添加导出任务到队列失败!");}}return AjaxResult.success();}
}

以下代码是通过定时任务拉取队列进行文件下载

@Slf4j
@Service
public class FileDownloadServiceImpl implements IFileDownloadService {@Resourceprivate ICustomerSettlementExportService customerSettlementExportService;@Resourceprivate IPubFileExportService pubFileExportService;@Resourceprivate RedisTemplate<String, String> redisTemplate;@Resourceprivate RedissonClient redissonClient;@Scheduled(fixedRate = 10000)public void processDownloadsInit() {// 提前过滤掉特定 IP 地址if (isLocalIp()) {return;}RLock lock = null;try {lock = redissonClient.getLock(BmsConstants.QUEUE_KEY_LOCK);boolean isOrderLock = lock.tryLock(10, TimeUnit.SECONDS);if (!isOrderLock) {// 当前任务未执行完,不能获取锁log.error("未能获取分布式锁,当前任务未执行完或有其他实例正在执行");return;}// 阻塞获取队列中的任务String exportTask = redisTemplate.opsForList().rightPop(BmsConstants.QUEUE_KEY, 10, TimeUnit.SECONDS);if (exportTask != null) {log.info("队列文件下载开始=" + DateUtils.getTime());handleDownload(exportTask);log.info("队列文件下载结束=" + DateUtils.getTime());}} catch (Exception e) {log.error("队列文件下载异常=", e);} finally {safeUnlock(lock);}}/*** 执行队列文件下载** @param exportTask 导出参数*/private void handleDownload(String exportTask) {try {PubFileExport pubFileExport = JSONObject.parseObject(exportTask, PubFileExport.class);// 实际的文件下载逻辑if (pubFileExport.getExportStartTime() == null|| pubFileExport.getExportEndTime() == null|| StringUtils.isBlank(pubFileExport.getClientCode())) {log.error("客户数据文件下载的参数为空请检查!");return;}AjaxResult ajaxResult = customerSettlementExportService.exportSheets(pubFileExport);pubFileExportService.updateExportAllExcel(ajaxResult, pubFileExport.getGid(), pubFileExport.getFinalUrl());} catch (Exception e) {log.error("执行队列文件下载异常=", e);}}/*** 检查是否为本地 IP 地址** @return 是否为本地 IP 地址*/private boolean isLocalIp() {try {InetAddress localAddress = InetAddress.getLocalHost();String ip = localAddress.getHostAddress();return ip.startsWith("192.168");} catch (Exception e) {log.error("获取本地 IP 地址异常=", e);return false;}}/*** 安全释放锁** @param lock 锁对象*/private void safeUnlock(RLock lock) {if (lock != null && lock.isLocked() && lock.isHeldByCurrentThread()) {try {lock.unlock();} catch (Exception e) {log.error("释放锁异常=", e);}}}}

3、解析下代码

这段代码实现了文件下载任务的队列管理和处理。
主要功能包括:
任务入队:将多个文件导出任务加入Redis队列。
定时任务处理:每隔10秒从队列中取出任务并执行文件下载。
分布式锁控制:确保同一时间只有一个实例处理任务,避免重复执行。
本地IP过滤:防止本地IP地址触发任务。

控制流图(CFG)

flowchart TD
A[开始] --> B{是否为本地IP}
B -->|是| E[结束]
B -->|否| C[获取分布式锁]
C --> D{是否获取成功}
D -->|否| F[记录日志并结束]
D -->|是| G[从队列取任务]
G --> H{是否有任务}
H -->|否| I[释放锁并结束]
H -->|是| J[处理下载任务]
J --> K[记录开始时间]
K --> L[调用下载逻辑]
L --> M[记录结束时间]
M --> N[释放锁并结束]

详细说明

任务入队 (pubFileExport 方法):
检查任务列表是否为空。
将每个任务序列化后加入Redis队列。
定时任务处理 (processDownloadsInit 方法):
检查是否为本地IP地址,如果是则直接返回。
尝试获取分布式锁,如果失败则记录日志并返回。
从Redis队列中取出任务,如果有任务则调用处理方法。
记录任务开始和结束时间,并释放锁。
处理下载任务 (handleDownload 方法):
反序列化任务参数。
检查参数是否完整,如果不完整则记录错误并返回。
调用客户结算导出服务进行文件下载,并更新导出状态。
辅助方法:
isLocalIp:检查当前IP是否为本地IP。
safeUnlock:安全释放分布式锁。

http://www.lryc.cn/news/523860.html

相关文章:

  • 第13章:Python TDD完善货币加法运算(二)
  • 两份PDF文档,如何比对差异,快速定位不同之处?
  • ESP-Skainet语音唤醒技术,设备高效语音识别方案,个性化交互应用
  • 地图:nuxt3高德地图简单使用 / nuxt2 + amap
  • 走进DevOps:让开发与运维齐头并进
  • 力扣动态规划-5【算法学习day.99】
  • LLM(3) : 浏览器录制16K的音频并上传到后端
  • PyTorch使用教程(13)-一文搞定模型的可视化和训练过程监控
  • 服务器日志自动上传到阿里云OSS备份
  • 树莓派学习
  • NestJS中实现注入多个实现了同一个接口的Service
  • Qt按钮美化教程
  • 基于单片机的多功能蓝牙语音智能台灯(论文+源码)
  • 第15章:Python TDD应对货币类开发变化(二)
  • 算法随笔_13: 有效三角形的个数
  • WSL 2 自动更新 虚拟 IP 到 window hosts
  • 我在广州学Mysql 系列——触发器的使用
  • 【useCallback Hook】在多次渲染中缓存组件中的函数,避免重复创建函数
  • 2025/1/20 学习Vue的第三天
  • Kotlin Bytedeco OpenCV 图像图像49 仿射变换 图像裁剪
  • 金融项目实战 07|Python实现接口自动化——连接数据库和数据清洗、测试报告、持续集成
  • (快速入门)保姆级详细的 Midjourney 基础教程
  • leetcode——找到字符串中所有字母异位词(java)
  • 大文件上传服务-后端V1V2
  • Single-Model and Any-Modality for Video Object Tracking——2024——cvpr-阅读笔记
  • 阳振坤:AI 大模型的基础是数据,AI越发达,数据库价值越大
  • Linux磁盘空间不足,12个详细的排查方法
  • Spring Web MVC综合案例
  • 微软预测 AI 2025,AI Agents 重塑工作形式
  • lvgl性能调优