【无标题】

旁路缓存笔记

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

逻辑:

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

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

具体逻辑:

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

命中:命中后直接缓存

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

先读后写问题

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

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

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

回写缓存失败问题

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

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

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

消息队列重试机制

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

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

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

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

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

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

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

相关推荐
木易 士心2 小时前
加密与编码算法全解:从原理到精通(Java & JS 实战版)
java·javascript·算法
专注于大数据技术栈2 小时前
java学习--ArrayList
java·学习
编程大师哥2 小时前
JavaEE初阶的核心组件
java·java-ee
sunnyday04262 小时前
深入理解分布式锁:基于Redisson的多样化锁实现
spring boot·redis·分布式
华如锦2 小时前
MongoDB作为小型 AI智能化系统的数据库
java·前端·人工智能·算法
EricLee2 小时前
2025 年终总结 - Agent 元年
前端·人工智能·后端
q***44152 小时前
C++跨平台开发挑战的技术文章大纲编译器与工具链差异
java·后端
JaguarJack2 小时前
PHP 8.5 升级生存指南:避免凌晨两点回滚的检查清单
后端·php·服务端
BingoGo2 小时前
PHP 8.5 升级生存指南:避免凌晨两点回滚的检查清单
后端·php