【无标题】

旁路缓存笔记

参考了小林coding和java全栈知识体系的文章

逻辑:

读:先读缓存,如果缓存中不存在就读数据库,然后更新缓存

写:先更新数据库,再删除缓存

具体逻辑:

失效:应用程序先从cache获取数据,如果没有得到,则从数据库中取数据数据,成功后放入缓存中

命中:命中后直接缓存

更新:先把数据库中的数据更新,然后删除缓存

先读后写问题

先读取cache发现为空,然后去db写入,随后一个写入请求写入db并删除cache,然后写操作再去cache写入。

这种情况会导致cache中存储的仍然是旧的数据。所以我们需要给缓存设置过期时间。

上面这个问题的发生的可能性不高,因为缓存写入的速度一般远远快于数据库更新+删除的速度。而且还有本身缓存过期时间的兜底,即使发生了先读后写导致的bug,也只是之前的数据没更新,等缓存的时间一过就可以访问了。

回写缓存失败问题

接下来还有问题:如果更新数据库+删除缓存这一套流程中,第二步删除缓存失败这个操作丢失怎么办?

通常来讲有两个解决方案:

1.消息队列重试机制 2. 订阅MySQL binlog,再操作缓存

消息队列重试机制

引入消息队列,将第二个操作要操作的数据加入到消息队列,让消费者来进行以下操作:

1.如果删除缓存失败,则进行重试,如果重试次数超过阈值,就返回给业务层报错

2.如果删除缓存成功,就要把数据从消息队列中移除。

这个方法有个问题就是对业务线有大的侵入。

异步更新缓存(基于订阅binlog的同步机制)

这个方案与消息队列的不同之处在于使用了数据库的binlog日志 ,然后订阅binlog来监控操作,一旦发现binlog中有更新的指令,就推送到非业务代码中执行前面的消息队列重试机制,好处就是因为监听的是数据库,所以对业务代码基本没有影响。

业界一般都是用的是binlog的异步更新缓存策略,可以使用阿里的中间件Canal来做binlog异步同步。

相关推荐
时空系几秒前
第6篇:数据容器——管理大量数据 Rust中文编程
开发语言·后端·rust
eLIN TECE7 分钟前
Go基础之环境搭建
开发语言·后端·golang
念何架构之路9 分钟前
Go反射应用技巧
开发语言·后端·golang
shjita10 分钟前
java根据键值对中值的大小进行排序的手法。
java·开发语言·servlet
司南-704928 分钟前
Dense结构下的 大模型系统架构研究
服务器·人工智能·后端
GISer_Jing28 分钟前
AI全栈转型_TS后端学习路线
前端·人工智能·后端·学习
薪火铺子29 分钟前
Spring Security 6.x 实战指南
java·后端·spring
全栈小刘33 分钟前
Claude Code 为什么这么顺?Anthropic 最新复盘:真正撑住它的不是模型,而是缓存
后端
敖正炀36 分钟前
WebFlux 深度:Reactor 线程模型、背压与错误处理
spring boot
BING_Algorithm39 分钟前
一文搞定 AOP 所有核心知识点
spring boot·后端·spring