缓存删除三级补偿方案:延迟队列+消息队列+定时任务兜底

问题背景:
在 Cache-Aside 模式中,更新数据库后删除缓存失败会导致数据不一致。本文提供工业级三级补偿方案,实现最终一致性保障。

整体架构:

复制代码
更新操作触发 → 一级延迟队列 → 二级消息队列 → 三级定时任务

方案实现:

一、第一级补偿:延迟队列(快速重试)

核心代码:

java 复制代码
// 延迟队列初始化
@PostConstruct
public void init() {
    deleteQueue = redisson.getQueue("cache:delete:queue");
    delayedQueue = redisson.getDelayedQueue(deleteQueue);
}

// 更新操作切面
@Around("@annotation(cacheUpdate)")
public Object aroundUpdate(...) {
    // 首次删除缓存
    redisson.getBucket(key).delete();
    
    // 数据库操作
    Object result = joinPoint.proceed();
    
    // 加入延迟队列(1秒后二次删除)
    delayedQueue.offer(key, 1, TimeUnit.SECONDS); 
    return result;
}

// 消费延迟队列(独立线程)
@EventListener(ApplicationReadyEvent.class)
public void startDelayConsumer() {
    new Thread(() -> {
        while (true) {
            String key = deleteQueue.poll(10, TimeUnit.SECONDS);
            if (key != null) {
                redisson.getBucket(key).delete();
            }
        }
    }).start();
}

特点:

响应时间:秒级

适用场景:高频更新业务

防抖设计:单线程顺序消费

二、第二级补偿:消息队列(可靠重试)

RocketMQ 集成示例:

java 复制代码
// 消息监听器
@RocketMQMessageListener(topic = "CACHE_DELETE", consumerGroup = "cache-group")
public void handleDelete(String key) {
    try {
        if (!redisson.getBucket(key).delete()) {
            retryWithBackoff(key, 3); // 指数退避重试
        }
    } catch (Exception e) {
        // 进入死信队列
    }
}

// 退避策略实现
private void retryWithBackoff(String key, int retryCount) {
    for (int i = 1; i <= retryCount; i++) {
        Thread.sleep(1000 * i);
        if (redisson.getBucket(key).delete()) break;
    }
}

// 延迟队列异常处理
delayedQueue.offer(...).exceptionally(e -> {
    rocketMQTemplate.send("CACHE_DELETE", key); 
    return null;
});

特点:

可靠性:99.9%+ 送达保障

重试策略:3次指数退避

容错机制:死信队列隔离异常

三、第三级补偿:定时任务(全量兜底)

Spring Scheduler 实现:

java 复制代码
// 每天凌晨执行全量比对
@Scheduled(cron = "0 0 3 * * ?") 
public void scanAndFix() {
    redisson.getKeys()
           .getKeysByPattern("user:*")
           .forEach(key -> {
               Long userId = extractUserId(key);
               User dbUser = databaseService.getUserFromDB(userId);
               User cacheUser = (User) redisson.getBucket(key).get();
               
               // 判断是否需要删除
               if ((cacheUser != null && dbUser == null) || 
                   (dbUser != null && cacheUser.getVersion() < dbUser.getVersion())) {
                   redisson.getBucket(key).delete();
               }
           });
}

比对策略:

缓存存在但数据库已物理删除

数据版本号不一致

逻辑删除标记状态不一致

防抖容错设计

  1. 防重复删除机制
java 复制代码
String deleteFlagKey = key + ":deleting";
if (redis.setnx(deleteFlagKey, "1")) {
    redis.expire(deleteFlagKey, 30); // 30秒窗口期
    performDelete(key); 
}
  1. 监控报警体系

各级补偿对比

相关推荐
ofoxcoding3 天前
在AI API聚合平台配置DeepSeek V3.2提示词缓存实战:快速接入与成本优化指南
人工智能·spring·缓存·ai
NeilYuen3 天前
gRPC结合FAISS构建AI助手语义缓存模块(一):设计
人工智能·缓存·faiss
taocarts_bidfans3 天前
反向海淘跨境缓存架构优化:taocarts Redis分层缓存实战技术
redis·缓存·架构·反向海淘·taocarts
退休倒计时3 天前
【每日一题】LeetCode 146. LRU 缓存 TypeScript
算法·leetcode·缓存·typescript
炘爚3 天前
Linux——Redis
数据库·redis·缓存
小挪号底迪滴3 天前
Redis 和 MySQL 数据不一致怎么办?缓存更新策略实战
redis·mysql·缓存
闪电悠米3 天前
黑马点评-Redis ZSet-实现关注 Feed 流
服务器·网络·数据库·redis·缓存·junit·lua
Saniffer_SH4 天前
【高清视频】Gen6 服务器还没到,Gen6 SSD 怎么测?Emily 现场演示三种测试环境
人工智能·驱动开发·测试工具·缓存·fpga开发·计算机外设·压力测试
AC赳赳老秦4 天前
OpenClaw + 飞书多维表格:自动同步数据、生成统计图表、触发自动化任务
java·大数据·python·缓存·自动化·deepseek·openclaw