首页 > 数据库 >怎样去保证 Redis 缓存与数据库双写一致性?

怎样去保证 Redis 缓存与数据库双写一致性?

时间:2024-03-28 13:04:04浏览次数:32  
标签:缓存 请求 删除 数据库 Redis 更新 双写

解决方案

那么我们这里列出来所有策略,并且讨论他们优劣性。

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

先更新数据库,后更新缓存

        这种方法是不推荐使用的,因为在更新缓存那一步有的业务需求缓存中的值并不是从数据库查,而是需要一系列计算而拿到缓存值,那这时候更新缓存的代价是非常大的。当有大量对数据库进行写的请求时,读的并不多,写一次请求就要更新一下缓存,那性能损耗真的很大,因为Redis是针对内存的。

        比如:当数据库有个值为100数值,我们有十个请求是要对其每次减五,这个期间完全没进行读的操作,那么先更新数据库,那会有十个请求对缓存更新,从而产生大量的冷数据,当不更新缓存而失去删除缓存,那么有读请求时只会更新一次。

先更新缓存,后更新数据库

这种方法不需要我们考虑了吧,和第一种方法是一样的。

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

该方法也会有问题,具体出现的原因:

这时来了两个请求,请求 A(更新) 和请求 B(查询)

  1. 请求 A 会先删除 Redis 中的数据,然后去数据库进行更新操作
  2. 此时请求 B 看到 Redis 中的数据时空的,会去数据库中查询该值,补录到 Redis 中
  3. 但是此时请求 A 并没有更新成功,或者事务还未提交

那么这时候就会产生数据库和 Redis 数据不一致的问题。如何解决:其实最简单的解决办法就是延时双删的策略。

上述的保证事务提交完以后再进行删除缓存还有一个问题,就是如果你使用的是 Mysql 的读写分离的架构的话,那么其实主从同步之间也会有时间差。

此时来了两个请求,请求 A(更新) 和请求 B(查询)

  1. 请求 A 更新操作,删除了 Redis
  2. 请求主库进行更新操作,主库与从库进行同步数据的操作
  3. 请 B 查询操作,发现 Redis 中没有数据
  4. 去从库中拿去数据
  5. 此时同步数据还未完成,拿到的数据是旧数据

解决办法就是如果是对 Redis 进行填充数据的查询数据库操作,就强制将其指向主库进行查询。

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

        这一种方法也会出现问题,当更新数据库成功了,但在删除缓存的阶段出错了没有删除成功,那此时再读取缓存的时候每次都是错误的数据了。

解决方案就是利用消息队列进行删除的补偿。具体的业务逻辑如下:

  1. 请求 A 先对数据库进行更新操作
  2. 在对 Redis 进行删除操作的时候发现报错,删除失败
  3. 此时将Redis 的 key 作为消息体发送到消息队列中
  4. 系统接收到消息队列发送的消息后再次对 Redis 进行删除操作

但是这个方案会有一个缺点就是会对业务代码造成大量的侵入,深深的耦合在一起,所以这时会有一个优化的方案,我们知道对 Mysql 数据库更新操作后再 binlog 日志中我们都能够找到相应的操作,那么我们可以订阅 Mysql 数据库的 binlog 日志对缓存进行操作。

总结        

        这些方法都有利弊,如在第二种先删除缓存,后更新数据库这个方法最后讨论了要更新 Redis 的时候强制走主库查询就能解决问题,那么这样的操作会对业务代码进行大量的侵入,但不需要增加的系统,不需要增加整体的服务的复杂度。最后一种方法我们最后讨论了利用订阅 binlog 日志进行搭建独立系统操作 Redis,这样的缺点其实就是增加了系统复杂度。

        所有的选择都需要我们对业务处理的评估来进行选择,没有一种技术是对所有业务通用的。没有最好的技术,只有最适合我们的。

标签:缓存,请求,删除,数据库,Redis,更新,双写
From: https://blog.csdn.net/m0_70770402/article/details/137107744

相关文章

  • 如何实现Redis集群的高可用性
    在实际应用中,确保Redis集群的高可用性是至关重要的。以下是一些常见的实现高可用性的方法和相关代码示例。1、主从复制(Master-SlaveReplication):原理:主节点负责处理数据写入操作,而从节点则从主节点复制数据。这样,即使主节点发生故障,从节点可以升级为主节点,继续提供服务......
  • Redis的相关配置
    #bind 127.0.0.1                   #注释掉这一句,使redis可以外部访问port 6379                         #默认端口,可以改成别的端口protected-mode yes                #修改为yes,开启保护模式,默认是yes#daem......
  • Redis 红锁:分布式锁的强大实现
    在分布式系统中,多个进程或线程可能需要并发访问共享资源。为了确保数据的一致性和正确性,我们需要一种分布式锁机制来协调这些访问。Redis红锁就是这样一种强大的分布式锁实现。一、分布式锁的概念分布式锁是一种用于在分布式系统中实现资源互斥访问的机制。它的目的是确......
  • Redis高可用之战:主从架构
    ★Redis24篇集合1主从模式介绍在笔者的另外两篇文章《Redis系列:RDB内存快照提供持久化能力》、《Redis稳定性之战:AOF日志支撑数据持久化》中,我们介绍了Redis中的数据持久化技术,包括RDB快照和AOF日志。有了这两个利器,我们再也不用担心机器宕机,数据丢失了。但是持久化技术......
  • window10 安装 Redis
    一、下载压缩包(免安装)地址:https://github.com/tporadowski/redis/releases二、解压解压路径自定义Logs目录,手动添加的,用于保存redis的日志记录(下面再补充)下图为解压后的文件三、注册Redis服务以管理员权限打开cmd窗口,执行以下命令#安装redis服务redis-server.ex......
  • iOS - Runtime - Class-方法缓存(cache_t)
    文章目录iOS-Runtime-Class-方法缓存(cache_t)1.散列表的存取值iOS-Runtime-Class-方法缓存(cache_t)Class内部结构中有个方法缓存(cache_t),用散列表(哈希表)来缓存曾经调用过的方法,可以提高方法的查找速度。(使用空间换时间的方式来提升速度)其中,_buckets就......
  • SpringBoot整合Redis:面试必考题-缓存击穿--逻辑过期解决
    ......
  • redis+aop实现接口防刷(幂等)
    幂等和接口防刷概念这两者其实是属于不同的场景但是在一些情况下,实现方式上有异曲同工之妙。防刷顾名思义,想让某个接口某个人在某段时间内只能请求N次。一般是对一些不发人员用脚本对接口进行大量请求,或者说利用脚本进行秒杀。幂等幂等的数学概念幂等是源于一种数学概念......
  • Redis常用命令
    1.StringSET将字符串值 value 关联到 key 。如果 key 已经持有其他值, SET 就覆写旧值,无视类型。>setkey01value01"OK">getkey01"value01"#设置value,同时设置key过期时间,单位秒>SETkey02value02EX60"OK">TTLkey02(integer)43SETEX将值......
  • http协议中缓存Cache-Control详解
    Cache-Control是一个HTTP/1.1协议中的头部字段,用于指定请求和响应遵循的缓存机制。通过这个头部,服务器可以告诉客户端响应可以被缓存多长时间,以及在什么条件下可以被缓存和重新使用。以下是一些常见的Cache-Control指令:基本指令no-cache表示响应不应该被缓存,除非得到......