Redis延迟双删实例详解(转)

在当前环境下,通常我们会首选redis缓存来减轻我们数据库访问压力。但是也会遇到以下这种情况:大量用户来访问我们系统,首先会去查询缓存, 如果缓存中没有数据,则去查询数据库,然后更新数据到缓存中,并且如果数据库中的数据发生了改变则需要同步到redis中,同步过程中需要保证 MySQL与redis数据一致性问题,在这个同步过程中出现短暂的数据延迟也是正常现象,但是最终需要保证mysql与缓存中的一致性。

//我们通常使用redis的逻辑
    //通常我们是先查询reids
    String value = RedisUtils.get(key);
    if (!StringUtils.isEmpty(value)){
        return value;
    }
//从数据库中获取数据
    value = getValueForDb(key);
    if (!StringUtils.isEmpty(value)){
           RedisUtils.set(key,value);
        return value;
     }

1、什么是延迟双删?

延迟双删策略是分布式系统中数据库存储和缓存数据保持一致性的常用策略,但它不是强一致。其实不管哪种方案,都避免不了Redis存在脏数据的问题,只能减轻这个问题,要想彻底解决,得要用到同步锁和对应的业务逻辑层面解决。

2、为什么要进行延迟双删?

一般我们在更新数据库数据时,需要同步redis中缓存的数据 所以我们一般会给出两种方案:

  • 第一种方案:先执行update操作,再执行缓存清除。
  • 第二种方案:先执行缓存清除,再执行update操作。

但是这两种方案在并发请求中容易出现以下问题

第一种方案弊端:当请求1去执行数据库更新操作之后,还没执行缓存清除时,请求2就进来了查询了缓存,此时缓存中数据还是旧数据,还没来得机删除导致数据出现问题,但是当t1执行缓存删除操作之后,后面的请求查询不到缓存,再到数据中查询,然后更新到缓存中,这种影响是比较小的

  1. t1线程 先更新db;
  2. t2线程查询命中缓存 返回旧的数据;
  3. 假设t1线程更新完db,预计5毫秒删除完缓存key 在5毫秒内 其他线程查询缓存结果还是为旧的数据,但是 5毫秒后查询缓存结果是为空,在从新将db最新的结果同步到Redis中。
  4. 一个项目中出现延迟是非常正常的,所以该情况发生的延迟对业务的影响其实很小。但是如果发生了,删除缓存失败呢?

1.不断重试----如果是在http协议接口中 会导致接口响应变慢 调用该接口 会发生响应超时 2.或者通过mq异步的形式同步

第二种方案弊端:当请求1执行清除缓存后,还未执行数据更新操作的时,请求2进来查询到数据库的旧数据,并写入了redis,这就导致了数据库与redis数据不一致问题。

  • t1线程先删除缓存;
  • t2线程读取缓存为null,同步db数据到缓存中;
  • t1线程更新db中的数据;
  • t3线程查询缓存中数据是旧数据;

3、对于方案处理都有弊端,那么我们需要使用延迟双删策略

先进行缓存清除,再执行update,最后(延迟N秒)再执行缓存清除。进行两次删除,且中间需要延迟一段时间

    RedisUtils.del(key);// 先删除缓存    updateDB(user);// 更新db中的数据    Thread.sleep(N);// 延迟一段时间,在删除该缓存key    RedisUtils.del(key);// 先删除缓存

4、需要注意的点

上述中(延迟N秒)的时间要大于一次写操作的时间。原因:如果延迟时间小于写入redis的时间,会导致请求1清除了缓存,但是请求2缓存还未写入的尴尬。。。

5、延迟的时间如何确定?

在业务程序运行时,统计业务逻辑执行读数据和写缓存的操作时间,以此为基础来进行估算。因为这个方案会在第一次删除缓存值后,延迟一段时间再次进行删除,所以称为“延迟双删”。

小结

延迟双删策略只是一种同步数据库与缓存的手段,在系统并发量不高的情况下可以使用这种方式解决,如果是并发量高的情况下我们也可以另寻其他解决方案 如:canal

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
下面是一个 Java 实现Redis 延迟双删的代码示例: ```java import redis.clients.jedis.Jedis; public class RedisDelayDoubleDelete implements AutoCloseable { private final Jedis jedis; private final String key; private final int expireTime; private Long deleteTime; public RedisDelayDoubleDelete(Jedis jedis, String key, int expireTime) { this.jedis = jedis; this.key = key; this.expireTime = expireTime; } @Override public void close() throws Exception { try { if (deleteTime != null) { long now = System.currentTimeMillis() / 1000; // 如果当前时间与上次删除时间相差不超过 expireTime,则进行延迟双删 if (now - deleteTime <= expireTime) { jedis.del(key); } } else { // 如果之前没有进行过删除,则记录当前时间 deleteTime = System.currentTimeMillis() / 1000; jedis.setex(key, expireTime, deleteTime.toString()); } } catch (Exception e) { // 如果发生异常,则不进行延迟双删 } } } ``` 使用方法如下: ```java try (RedisDelayDoubleDelete redisDelayDoubleDelete = new RedisDelayDoubleDelete(jedis, key, expireTime)) { // 执行一些需要保证幂等性的操作 } // RedisDelayDoubleDelete 实例的 close 方法会检测是否需要进行延迟双删,并在必要时执行 ``` 在上面的代码中,`RedisDelayDoubleDelete` 类实现了 `AutoCloseable` 接口,它在 `close` 方法中检测是否需要进行延迟双删,如果需要则删除 Redis 中的键,并记录删除时间。在使用时,可以使用 `try-with-resources` 语句来自动调用 `close` 方法。如果发生异常,则不进行延迟双删

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值