【无标题】

旁路缓存笔记

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

逻辑:

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

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

具体逻辑:

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

命中:命中后直接缓存

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

先读后写问题

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

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

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

回写缓存失败问题

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

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

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

消息队列重试机制

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

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

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

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

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

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

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

相关推荐
侠客行03175 小时前
Mybatis连接池实现及池化模式
java·mybatis·源码阅读
蛇皮划水怪5 小时前
深入浅出LangChain4J
java·langchain·llm
Victor3565 小时前
https://editor.csdn.net/md/?articleId=139321571&spm=1011.2415.3001.9698
后端
Victor3565 小时前
Hibernate(89)如何在压力测试中使用Hibernate?
后端
灰子学技术7 小时前
go response.Body.close()导致连接异常处理
开发语言·后端·golang
老毛肚7 小时前
MyBatis体系结构与工作原理 上篇
java·mybatis
风流倜傥唐伯虎7 小时前
Spring Boot Jar包生产级启停脚本
java·运维·spring boot
Yvonne爱编码7 小时前
JAVA数据结构 DAY6-栈和队列
java·开发语言·数据结构·python
Re.不晚7 小时前
JAVA进阶之路——无奖问答挑战1
java·开发语言
你这个代码我看不懂7 小时前
@ConditionalOnProperty不直接使用松绑定规则
java·开发语言