欢迎访问个人博客: https://www.crystalblog.xyz/
备用地址: https://wang-qz.gitee.io/crystal-blog/
见第4点先更新db再删除缓存 or 先删除缓存再更新db
的详解.
见第4点先删除缓存再更新db
中的延迟双删方案
见第4点先删除缓存再更新db
中的方案选择
先更新db, 再删除缓存, 会结合MQ异步删除缓存, 延迟比较高.
伪代码:
updateDB(user); // 更新db数据
sendMqMsgToDeleteRedis(key); // 先删除缓存
如果更新db成功, 但是删除缓存失败如何处理? MQ重试机制, 保证该message消息必须消费成功.
先删除缓存, 再更新db, 延迟低.
伪代码:
deleteRedis(key); // 先删除缓存
updateDB(user); // 更新db数据
如果A请求删除缓存成功, 但是更新db失败, 其他请求过来又重新查询db的旧数据存入到Redis缓存中, 会导致Redis与mysql数据一致性问题. 如何解决呢, 初始解决方案—延迟双删
延迟双删方案
T1线程线删除缓存再更新db , T1线程更新db完成之前T2线程如果读取到db旧的数据, 会再把旧的数据写入Redis缓存, 此时T1线程延迟一段时间后再删除Redis缓存操作. 当其他线程再读取缓存为null时会查询db最新数据重新进行缓存, 保证了Mysql和Redis缓存的数据一致性.
所以, T1线程sleep的时间, 就需要大于T2线程读取数据再写入Redis缓存的时间.
延迟时间如何确定?
在业务程序运行时, 统计业务逻辑执行读取数据和写Redis缓存的操作时间, 以此为基础进行估算. 因为这个方案会在第一次删除缓存值后, 延迟一段时间再次进行删除, 所以称为"延迟双删" . — 不推荐大家使用延迟双删.
伪代码:
deleteRedis(key); // 先删除缓存
updateDB(user); // 更新db数据, 事务未完成
sleep(seconds); // 延迟一段时间,待其他线程读取db旧数据缓存到Redis完成
deleteRedis(key); // 再删缓存
延迟双删方案方案如何选择?
绝大多数场景都会将Redis作为只读缓存:
(1) 先删除缓存, 再更新db. 更新db完成之前, 其他线程在读取缓存时发现数据为null, 会将旧数据重新缓存到Redis中, 导致其他线程一直查询的数据为旧的数据. 需要考虑延迟双删问题, 延迟双删的时间不是很好控制.
(2) 也可以先更新db再结合MQ异步删除Redis缓存; 因为先更新db, 再删除缓存, 其他线程在读过程中只是短暂读取到数据(旧数据), 只要及时将该缓存key删除, 其他线程就可以读取到最新的数据.
所以推荐使用先更新db, 再删除缓存.
更新完db后, 同步更新Redis缓存; 更新db操作与更新Redis缓存是一样的, 不是直接删除Redis缓存key, 俗称 “双写”.
t1线程先更新db, 再更新Redis缓存; 如果t1线程在更新Redis缓存完成之前, t2线程优先完成Redis缓存的更新, 然后t1线程再完成Redis缓存的更新, 导致db数据xiaojun
和Redis缓存的数据mayikt
不一致问题.
导致不一致的时间线:
t1线程更新db – t2线程更新db – t2线程更新缓存 – t1线程更新缓存.
理想结果的时间线:
t1线程更新db – t1线程更新缓存-- t2线程更新db – t2线程更新缓存.
初始解决方案:
可以将更新db的事务放到更新完缓存后提交(Mysql行锁保证数据一致性). 但是性能很低.
伪代码:
begin(transaction); // 开启事务
updateDB(user); // 更新db
updateRedis(key); // 更新缓存
commit(transaction); // 提交或回滚事务
进阶解决方案:
zookeeper等分布式锁.
最终解决方案:
推荐MQ异步双写, 没有锁的竞争, 效率比较高. 但是会有延迟问题. 能够保证最终一致性.
可以使用canal + kafka + 消息顺序一致性
见第6点的并发的情况下如何保证双写一致性?
中的进阶解决方案
见第6点的并发的情况下如何保证双写一致性?
中的初始解决方案
保证最终一致性, 见第6点的并发的情况下如何保证双写一致性?
中的最终解决方案