缓存数据和数据库数据一致性问题

根据以上的流程没有问题,但是当数据变更的时候,如何把缓存变到最新,使我们下面要讨论的问题

1. 更新数据库再更新缓存

场景:数据库更新成功,但缓存更新失败。

问题

  • 当缓存失效或过期时,读取到的可能是旧数据。这样会导致读取到的不一致数据,从而影响系统的正确性和用户体验。
2. 更新缓存再更新数据库

场景:缓存更新成功,但数据库更新失败。

问题

  • 此时,读取的缓存可能是错误的,因为数据库的状态并未同步到缓存。这会导致后续的数据查询不准确,从而影响业务逻辑。
3. 先删除缓存再更新数据库

优点

  • 这种方式在一定程度上缓解了数据不一致的问题。即使数据库更新失败,缓存也会被清空。

问题

  • 高并发场景下的问题
    • 假设有两个线程 A 和 B:
      • A 线程删除了缓存并正在更新数据库。
      • B 线程发现缓存已被删除,便去查询数据库并将旧数据放入缓存。
    • 结果是,B 线程可能会将过时的数据写入缓存,导致后续的请求获取到过期数据。
延迟双删策略

为了解决上述问题,可以采用延迟双删的策略:

  1. 步骤

    • 在删除缓存后,更新数据库。
    • 数据库更新成功后,再次删除缓存。
  2. 目的

    • 第一次删除缓存是为了确保下次请求能重新加载最新数据。
    • 第二次删除是为了确保即使在高并发情况下,B 线程获取到的旧缓存也能被及时清除(但延迟双删存在一个潜在问题:在第二次删除缓存的过程中,如果有新的线程访问缓存,这些线程仍然有可能获取到旧数据,导致系统状态不一致)
扩展思路
  1. 消息队列补偿机制

    • 当缓存删除失败时,可以将失败信息作为消息发送到消息队列(MQ)。
    • MQ 消费者监听这些消息,并进行重试,确保缓存得到正确的更新。
    • 优点:异步处理,提高系统的可用性与稳定性。
  2. Canal

    • 通过 Canal 等工具监听数据库的变化。
    • 将数据库变更事件发送到缓存刷新服务。
    • 优点:业务逻辑解耦,减少冗余代码,提高系统的复杂度可管理性。
相关推荐
卜及中37 分钟前
【Redis/2】核心特性、应用场景与安装配置
数据库·redis·缓存
LucianaiB1 小时前
如何做好一份优秀的技术文档:专业指南与最佳实践
android·java·数据库
Eiceblue1 小时前
Python读取PDF:文本、图片与文档属性
数据库·python·pdf
敖云岚4 小时前
【Redis】分布式锁的介绍与演进之路
数据库·redis·分布式
LUCIAZZZ4 小时前
HikariCP数据库连接池原理解析
java·jvm·数据库·spring·springboot·线程池·连接池
我在北京coding5 小时前
300道GaussDB(WMS)题目及答案。
数据库·gaussdb
小Tomkk5 小时前
阿里云 RDS mysql 5.7 怎么 添加白名单 并链接数据库
数据库·mysql·阿里云
明月醉窗台6 小时前
qt使用笔记二:main.cpp详解
数据库·笔记·qt
沉到海底去吧Go6 小时前
【图片自动识别改名】识别图片中的文字并批量改名的工具,根据文字对图片批量改名,基于QT和腾讯OCR识别的实现方案
数据库·qt·ocr·图片识别自动改名·图片区域识别改名·pdf识别改名
老纪的技术唠嗑局7 小时前
重剑无锋,大巧不工 —— OceanBase 中的 Nest Loop Join 使用技巧分享
数据库·sql