老话新谈之缓存一致性

news/2025/1/12 19:04:06/

前言

缓存一致性常见的更新策略也比较多,如先更新数据库再更新缓存,先删缓存再更新数据库等等,我在理解的时候有些混乱,所以这个文章提供了一些理解上的技巧去理解缓存一致性。

为什么会有缓存一致性的问题

  1. 缓存与数据库是两套中间件,存在网络抖动之类的原因导致没有更新任一方的可能
  2. 数据库大多都是事务型的中间件,支持错误回滚,缓存大多是非事务型的中间件,这里缓存更新失败了没办法回滚

所以根因是缓存大部分不支持事务无法回滚。

怎么尽量解决缓存一致性的问题

操作二者必定有先后顺序,存在以下两个情况:

  1. 先操作缓存,再操作数据库。操作缓存成功,数据库更新失败,缓存无法回滚,数据不一致
  2. 先操作数据库,再操作缓存。操作数据库成功,缓存操作失败,可触发异常回滚数据库,数据一致

根据上述所列,只能先操作数据库,再操作缓存了。

操作缓存也分两种:

  1. 更新缓存数据,可能并发请求,后一请求更新缓存的数据被前一请求的更新覆盖了,导致数据不一致
  2. 删除缓存数据,并发请求,二者都使缓存失效,查询请求将数据库数据加载到缓存中,数据一致

根据上述所列,只能使缓存失效,查询请求加载数据到缓存中了。

所以,如果在不加任何重试措施的情况下,先操作数据库,再删除缓存是一个容错较好的方法。

缓存一致性的分类 & 存在的问题

Client 维护缓存 & 数据库的一致性

  1. 更新缓存 -> 更新数据库

    图片1

@startuml
Database Database   as DB
entity   Cache      as Cachetransaction1 -> Cache: update data
transaction1 <-- Cache: update resulttransaction1 -> DB: update data
transaction1 <-- DB: update result@enduml
  • 可能出现的数据不一致

​ 数据不一致:更新缓存成功了,更新数据库失败了,有数据不一致的问题,直到缓存超时失效或又一更新请求操作成功都会不一致

  • 改进方式

    若保证更新数据仅有少数的服务更新,可以将更新数据库请求入队处理,且可加入重试机制。但是队列的加入会增大系统复杂度,并且重试以及缓存更新顺序不一致会加剧数据不一致

  1. 更新数据库 -> 更新缓存

@startuml
Database Database   as DB
entity   Cache      as Cachetransaction1 -> DB: update data
transaction1 <-- DB: update resulttransaction2 -> DB: update data
transaction2 <-- DB: update resulttransaction2 -> Cache: update data
transaction2 <-- Cache: update resulttransaction1 -> Cache: update data
transaction1 <-- Cache: update result@enduml
  • 可能出现的数据不一致

​ 数据不一致:如 t1 先更新数据库,t2 在 t1 更新缓存前把数据库缓存都更新完了,t1 再更新缓存,这时候缓存上是 t1 的数据,数据库是 t2 的数据

  • 改进方式

    若保证更新数据仅有少数的服务更新,可以将更新数据库请求入队处理,但是队列更新的引入增大了系统复杂度

  1. 删除缓存 -> 更新数据库

@startuml
Database Database   as DB
entity   Cache      as Cachetransaction1 -> Cache: delete dataquery1 -> DB: select data
query1 -> Cache: insert datatransaction1 -> DB: update result@enduml
  • 可能出现的数据不一致

    1. 如图所示,更新请求先删除缓存,查询请求从缓存获取不到数据从数据库获取数据(老数据)加载到缓存中,更新请求更新数据库
    2. 这样的流程会导致查询请求加载老数据到缓存中,后续更新请求更新新数据到数据库中,导致数据不一致
  • 改进方式

    暂无。

  1. 更新数据库 -> 删除缓存

@startuml
Database Database   as DB
entity   Cache      as Cachequery1 -> DB: select data
transaction1 -> DB: update result
transaction1 -> Cache: delete data
query1 -> Cache: insert data@enduml
  • 可能出现的数据不一致

    查询请求先拿到数据,在插入缓存前更新请求进来更新数据库并使缓存失效,这个请求比较罕见

    1. 发生的场景
      1. 查询请求所在机器请求缓存比更新请求做完的整个流程都要慢
    2. 发生的概率
      1. 很低。因为操作缓存一般会比操作数据库要快
  • 改进方式

    1. 变更数据记录变更事件
      1. 步骤
        1. 更新数据同步记录一个事件在本地内存中
        2. 查询请求在插入缓存前查询事件,如果存在变更则查数据库获取最新数据
        3. 如果此数据在查询请求插入缓存过程中一直变更,这里需要先返回当前数据库结果给上游,再开异步任务轮训事件/数据库插入缓存
      2. 适用场景
        1. 只适用单节点

Server 维护缓存 & 数据库的一致性

  1. Read though/Write though

    • read though

    @startuml
    Database Database   as DB
    entity   Cache      as Cachequery -> repository: select datarepository -> cache: get data
    repository -> DB: get data
    DB -> repository: return data
    repository -> cache: update data
    repository -> query: return data@enduml
    
    • wirte though

    @startuml
    Database Database   as DB
    entity   Cache      as Cachetranscation -> repository: update datarepository -> cache: update data
    repository -> DB: update data
    DB -> repository: return result
    repository -> transcation: return result@enduml
    
  • 可能出现的数据不一致
    • 程序没有优雅关闭,更新请求先更新了缓存,但还没更新数据库,数据丢失
    • 更新缓存成功,更新数据库失败导致的数据不一致
  • 适用场景
    • 更新数据库极低概率失败
    • 程序有优雅关闭功能
  • 改进方式
    • 暂无
  1. Write Behind

@startuml
Database Database   as DB
entity   Cache      as Cachequery -> repository: query datarepository -> cache: query data
repository -> DB: query data
DB -> repository: return data
repository -> cache: update datarepository -> query: return data@enduml

@startuml
Database Database   as DB
entity   Cache      as Cachetranscation -> repository: update datarepository -> cache: update datarepository -> DB: batch update data@enduml
  • 可能出现的数据不一致
    • 程序没有优雅关闭,更新请求先更新了缓存,但还没更新数据库,数据丢失
    • 批量更新数据库失败导致的数据不一致
  • 适用场景
    • 更新数据库极低概率失败
    • 程序有优雅关闭功能
  • 改进方式
    • 暂无

参考

https://coolshell.cn/articles/17416.html

本文首发于cartoon的博客

转载请注明出处:https://cartoonyu.github.io


http://www.ppmy.cn/news/314821.html

相关文章

09.JavaWeb-MyBatis

3.MyBatis MyBatis 是一款优秀的持久层框架&#xff0c;它支持自定义 SQL、存储过程以及高级映射。MyBatis 免除了几乎所有的 JDBC 代码以及设置参数和获取结果集的工作。 3.1 配置MyBatis 3.1.1 导入依赖 <dependency><groupId>mysql</groupId>&l…

iphone ipad 上传图片到oss上失败。

我遇到的问题原因是&#xff1a; 使用的是http上传。需要使用https上传。 备注&#xff1a;&#xff08;苹果IOS要求在2017年1月1日&#xff0c;将强制启用ATS。App Transport Security&#xff0c;简称 ATS&#xff0c;是苹果在 iOS 9 当中首次推出的一项安全功能。在启用 ATS…

vs2017开发android平板,2018 iPad vs. 2017 iPad跑分对比

外媒appleinsider为我们带来了新款iPad跟去年老款的跑分对比&#xff1a; 结果显示&#xff0c;在Geekbench 4 单核测试中&#xff0c;搭载了A10 Fusion芯片的2018 iPad其运行速度比搭载了A9的2017 iPad快了44%&#xff0c;多核的话则更高&#xff0c;达到了53%。 不过在GPU测试…

ipad怎么和mac分屏_如何将 iPad 作为 Mac 的扩展屏幕 |「随航(Sidecar)」功能详解...

今日,苹果推送了 macOS 10.15 Catalina 正式版,其中,随航(Sidecar)是伴随 iPadOS 而来的一个全新功能,通过此功能可以将 iPad 随时变成 Mac 的扩展屏幕。 当设备更新到 macOS 10.15 以及 iPadOS 13.1 以上后,可以通过点击顶部 AirPlay 选项中选择「打开 Sidecar 偏好设置」…

python ipados_macOS Big Sur 正式发布并已开放下载,支持原生运行 iOS 和 iPadOS App

SegmentFault 思否消息&#xff0c;11月13日凌晨&#xff0c;macOS Big Sur 正式发布并已开放下载&#xff0c;版本号为 macOS 11.0.1 (20B29)&#xff0c;大小 12.6G。支持原生运行 iOS 和 iPadOS App。 这次 macOS Big Sur 在 UI 设计上有了较大幅度的改动&#xff0c;整体风…

ipad出现support.apple.com

前几天ipad无缘无故白屏重启&#xff0c;无奈选择更新系统&#xff0c;但是更新后出现如下问题&#xff1a; Support apple.com/iphone/restore的提示。 步骤一&#xff1a;在电脑下载并安装iTunes&#xff1b; 步骤二&#xff1a;将苹果手机与电脑连接&#xff1b; 步骤三&…

ipadmini5可以安装eclipse嘛_MyEclipse 2017软件安装教程

MyEclipse2017(32/64位)下载地址&#xff1a; 链接&#xff1a;https://pan.baidu.com/s/1c4i0J5Y 密码&#xff1a;hqfw MyEclipse2017(MAC)下载地址&#xff1a; 链接: https://pan.baidu.com/s/1qXF5qZM 密码: mkt2 MyEclipse2017(Linux)下载地址&#xff1a; 链接: https…

python ipad pro_离开 PC,在 iPad Pro 上也能编程了?

论基于 iPad Pro 平台编码的可行性。 作者 |Andrew Brookins 译者 |弯月 责编 | 屠敏 出品 |CSDN&#xff08;ID&#xff1a;CSDNNews&#xff09; 2017年我曾提出过一个问题&#xff0c;“你能在iPad上写代码吗&#xff1f;”&#xff08;https://andrewbrookins.com/tec…