当前位置: 首页 > news >正文

解决MySQL与Redis缓存一致性的问题

背景

考试系统中,教师会在后台发布一场考试,考试会存储在MySQL和Redis里面,考试有时候是会出错的,我们需要后台修改,如果多个教师在后台并发修改(概率不大),可能会出现数据库缓存不一致的问题,我们需要解决一下,刚好联系一下这一块的知识

基础知识理论

四种解决办法:

  1. 先更新数据库,再更新缓存
  2. 先更新缓存,后更新数据库
  3. 先删除缓存,后更新数据库
  4. 先更新数据库,后删除缓存

这四种情况,都会出现缓存不一致的问题,但是第四种出现缓存不一致的情况概率上比较低,因为缓存的写入回远远快于数据库的写入,我们可以使用【先更新数据库+再更新缓存】的方式来解决数据库不一致的问题,为了确保万无一失,我们还可以加上一个过期时间。

但是有两个问题

  1. 我们需要保证先更新数据库+再更新缓存是一个原子操作,不然如果更新缓存失败了,导致缓存中的数据还是旧值
  2. 「先更新数据库,再删除缓存」的方案虽然保证了数据库与缓存的数据一致性,但是每次更新数据的时候,缓存的数据都会被删除,这样会对缓存的命中率带来影响。

对于第一个问题,解决办法是消息队列重试机制、订阅MySQL binlog,再操作缓存

对于第二个问题是,我们使用「更新数据库 + 更新缓存」的方式来实现,但是这个方案本身固有一些问题,解决办法是,分布式锁和过期时间

另外先删除缓存,再更新数据库的解决方案是延迟双删

实战-先更新数据库,后删除缓存

此种方案存在问题是两个操作不是原子性的,如何保证两个操作都能成功呢?

方案一:gin框架+消息队列

我们实现一个中间件

func Canal() gin.HandlerFunc {//封装成一个中间件,当我们更新考试的时候,会自动监测到return func(c *gin.Context) {c.Next()redisDB := global.GVA_REDISget, exists := c.Get("DeleteRedisKey")DeleteKey := get.(string)if exists {err := redisDB.Del(context.Background(), DeleteKey).Err()if err == redis.Nil {//无需删除c.Next()} else if err != nil {//删除失败,加入到redis异步队列中,再次删除addToDelayedQueue(context.Background(), redisDB, DeleteKey, time.Now().Add(10*time.Second))}else {global.GVA_LOG.Info("更新数据成功")}} else {zap.L().Info("redis中不存在该键,无需删除")}}
}

我们在继承这个中间件的接口中的使用c.Set方法传递要删除的redisKey,然后我们在上述中间件中使用c.Get方法来接收要删除的redisKey,然后加入到异步队列中删除

现在来模拟一下是否能够成功,项目启动之后,把redis停机,然后删除redis的key就会失败,然后该key就会加入延时队列

方案一:消息队列订阅binlog

里面有两个难点,一个是订阅binlog服务,一个是消息队列

 如何订阅binlog服务,我们使用阿里巴巴的canal ,支持Go语言,具体参考Golang通过alibabaCanal订阅MySQLbinlog_大杯无糖的博客-CSDN博客

接下来是用redis的zset实现一个异步消息队列

http://www.lryc.cn/news/121334.html

相关文章:

  • 王道机组难题分析
  • 数学建模(一)前继概念
  • C# 随机法求解线性规划问题 蒙特卡洛
  • nginx文档合集
  • 什么是BFC?它有什么作用?如何创建BFC?
  • svn文章四:版本控制策略 - 穿越时光机:SVN版本控制进阶技巧
  • SpringBoot+Mybatis-Plus实现增删改查超详细步骤
  • Qt应用开发(基础篇)——拆分器窗口 QSplitter QSplitterHandle
  • 屏幕尺寸单位 px、em、rem区别
  • yo!这里是STL::list类简单模拟实现
  • 小程序商城开发制作
  • 并发编程面试题2
  • 关于eclipse导入部署具有增删改查的项目
  • c++日志工具之——log4cpp
  • ES索引重建reindex详解
  • 前沿分享-中距离射频取电
  • UnrealEngine - 网络同步之连接篇
  • 【JDBC系列】- 扩展提升学习
  • 阻塞和非阻塞,同步和异步
  • 提速Rust编译器!
  • QT创建项目
  • 基于vue3+webpack5+qiankun实现微前端
  • 华为OD真题--完美走位--带答案
  • 【AI】《动手学-深度学习-PyTorch版》笔记(十四):多层感知机
  • 本地开发 npm 好用的http server、好用的web server、静态服务器
  • Gradio入门,并搭个鸡兔同笼问题小应用,附源码(MindOpt)
  • redis核心知识点简略笔记
  • 消息中间件 —— 初识Kafka
  • Ceph集群安装部署
  • PXC基于docker搭建mysql集群全过程