go-zero微服务实战系列(六、缓存的一致性如何保证)

只要我们使用缓存,就必然会面对缓存和数据库间的一致性问题。如果缓存中的数据和数据库的数据不一致,那么业务应用从缓存中读取的数据就不是最新的数据,对业务的影响可想而知。比如我们把商品的库存数据存在缓存中,如果缓存中库存数据不对,那么可能就会影响下单操作,这是业务上很难接受的。本篇文章我们来一起聊一聊缓存的一致性问题。

如何解决缓存不一致#

先删缓存再更新数据库#

假设线程 A 删除缓存后,还没来得及更新数据库,这时候线程 B 开始读数据,线程 B 发现缓存缺失就只能去读数据库,等到线程 B 从数据库中读取完数据回塞缓存后,线程 A 才开始更新数据库,此时,缓存中的数据是旧值,而数据库中是最新值,两者已经不一致了。

这种场景的解决方案是在线程 A 更新完数据库的值后,可以让它 sleep 一小段时间,再进行一次缓存删除操作,之所以要加上 sleep 的一段时间,就是为了让线程 B 能够先从数据库读取出数据然后再把缓存 miss 的数据回塞到缓存,然后线程 A 再进行删除。所以线程 A 的 sleep 时间就需要大于线程 B 读取数据再写入缓存的时间。这个时间是多少呢?这个是需要我们在业务中加入打点监控来统计的,根据这个统计值来估算该时间。这样一来,其他线程读取数据时,会发现缓存缺失,就会从数据库中读取最新的值。我们把这种模型叫做 “延时双删”。

先更新数据库再删除缓存#

如果线程 A 更新了数据库中的值,但还没来得及删除缓存中的值,线程 B 这时候开始读取数据,此时,线程 B 查询缓存时,命中了缓存,就会直接使用缓存中的值,该值为旧值。不过在这种场景下,如果并发请求量不高的话,其实基本上不会有线程读到旧值,而且线程 A 更新完数据库后,删除缓存是非常快的操作,所以,这种情况总体对业务影响较小。一般在生产环境中,也推荐大家采用该模式。

重试机制#

可以把要删除的缓存值或者要更新的数据库的值放到消息队列中,当应用没能够成功地删除缓存或者是更新数据库的值的时候,可以从消息队列中消费这些值,这里消费消息队列的服务叫 job,然后再次进行删除或者更新,起到一个兜底补偿的作用,以此来保证最终的一致性。

如果能够成功地删除或更新,就需要把这些值从消息队列中去除,以免重复操作,此时,我们也可以保证数据库和缓存数据的一致了,否则的话,我们还需要再次进行重试,如果重试超过一定次数还是失败,这时候一般都需要记录错误日志或者发送告警通知。

并发读写#

首先第一步线程 A 读取缓存,这时候缓存没有命中,由于使用的是 cache aside 这种模式,所以接下来第二步线程 A 会去读数据库,这个时候线程 B 更新数据库,更新完数据库后通过 set cache 更新了缓存,最后第五步线程 A 把从数据库读到的值通过 set cache 也更新了缓存,但是这时候线程 A 中的数据已经是脏数据了,由于第四步和第五步都是设置缓存,导致写入的值相互覆盖,并且操作的顺序具有不确定性,从而导致了缓存不一致情况的发生。

怎么解决这个问题呢?其实非常地简单,我们只需要把第五步的 set cache 操作替换成 add cache 即可,add cache 即 setnx 操作,只有缓存不存在的时候才会成功写入,相当于加了优先级,即更新数据库后的更新缓存优先级更高,而读数据库后回塞缓存的优先级较低,从而保证写操作的最新数据不会被读操作的回塞数据覆盖。

结束语#

本篇文章说明了在使用缓存时最常遇见的一个问题,也就是缓存和数据库不一致的问题,针对这个问题我们列举了一些可能导致不一致的场景以及对应场景的解决方案,特别地,对于 job 异步补偿的场景我们可以使用 set 操作来强行覆盖缓存,保证缓存的更新为最新的数据,而对于读数据库回塞缓存的操作我们一般使用 add 来更新缓存。

希望本篇文章对你有所帮助,谢谢。

每周一、周四更新

代码仓库: github.com/zhoushuguang/lebron

项目地址#

github.com/zeromicro/go-zero

欢迎使用 go-zerostar 支持我们!

微信交流群#

关注『微服务实践』公众号并点击 交流群 获取社区群二维码。

本作品采用《CC 协议》,转载必须注明作者和本文链接
kevwan
讨论数量: 1

如果采用旁路缓存,文章中讨论的并发读写是不是不存在了?因为写操作更新数据的时候是删缓存而不存在设置缓存

7个月前 评论

go-zero作者 @ 某互联网公司
文章
102
粉丝
639
喜欢
651
收藏
619
排名:151
访问:6.6 万
私信
所有博文
社区赞助商