Redis缓存数据一致性

2023-12-23 18:52

本文主要是介绍Redis缓存数据一致性,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

实际业务中常使用Redis缓存来提升读写效率,减少存储层的压力。因为数据在缓存和DB中各存储一份,所以会出现数据一致性的问题。总体来说导致数据不一致的原因主要有两个。请求并发操作非原子
请求并发是指同时可能有多个读写请求同时请求Cache或者DB,前后乱序,最终导致缓存/DB数据更新不一致。
同时操作Cache和DB属于两个操作,两个非原子性操作,如果出现第二步失败,同样也会出现数据的不一致。

请求并发

缓存处理方式

对Cache数据的处理方式有两种,分别为更新删除。哪种方式更能保证数据一致性呐?接下来一个个来分析。

更新缓存

使用更新Cache的方式,无论是先更新Cache还是后更新Cache,都有可能出现数据不一致问题。
举例说明,假如并发A、B两个写请求。A将数据X更新为1,B将数据更新为2。其中B更新晚于A。

先更新Cache,后更新DB

最终结果Cache=2,DB=1
先更新Cache后更新DB

先更新DB,后更新Cache

最终结果Cache=1,DB=2
先更新DB后更新Cache

删除缓存
先删除Cache,后更新DB

举例说明,假如并发A、B两个请求。A为写请求,B为读请求。
先删除cache后更新DB

先更新DB,后删除Cache

举例说明,假如并发A、B两个请求。A为读请求,B为写请求。
先更新DB,后删除Cache的方式可以看出也会存在一致的场景。但是仔细思考下,出现这种场景的概率非常低。他需要满足A读请求写Cache的耗时大于B请求更新数据库+删除Cache的耗时。因为DB操作的耗时要远大于Cache的操作耗时。所以先更新DB,后删除Cache出现数据不一致的问题出现几率比较小。
先更新DB后更新Cache

总结

综上,针对cache处理方式和操作顺序进行分析,最终最靠谱的方案就是先更新DB,后删除Cache
综合分析
但是如果业务场景要求为弱一致性或者最终一致性。先删除Cache后更新DB的方式也是可以接受的,同时安全起见,可以引入延时双删的策略。在写请求更新完DB后休眠一会儿,再次将缓存删除,可以达到最终一致性的要求。

操作非原子

上面讨论的情况都是在两步操作都成功的前提下进行的。但是如果第一步操作成功,但是第二步操作失败,那么无论是Cache数据如何操作都会导致数据的不一致。
那么如何保证两步操作都保证成功,最简单的解决方案就是重试。重试的方案最为常见的就是引入MQ,进行异步重试。以先更新DB,后删除缓存方式为例。如果在第二步操作删除缓存失败时,可以发通知给MQ,然后通过消费者接受通知消息然后重试删除Cache。
MQ消息异步重试
另外除了MQ方案,还有订阅binlog的方案。在DB数据发生变更时,通过订阅binlog在更新/删除Cache。

这篇关于Redis缓存数据一致性的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/529102

相关文章

零基础学习Redis(10) -- zset类型命令使用

zset是有序集合,内部除了存储元素外,还会存储一个score,存储在zset中的元素会按照score的大小升序排列,不同元素的score可以重复,score相同的元素会按照元素的字典序排列。 1. zset常用命令 1.1 zadd  zadd key [NX | XX] [GT | LT]   [CH] [INCR] score member [score member ...]

Redis中使用布隆过滤器解决缓存穿透问题

一、缓存穿透(失效)问题 缓存穿透是指查询一个一定不存在的数据,由于缓存中没有命中,会去数据库中查询,而数据库中也没有该数据,并且每次查询都不会命中缓存,从而每次请求都直接打到了数据库上,这会给数据库带来巨大压力。 二、布隆过滤器原理 布隆过滤器(Bloom Filter)是一种空间效率很高的随机数据结构,它利用多个不同的哈希函数将一个元素映射到一个位数组中的多个位置,并将这些位置的值置

Lua 脚本在 Redis 中执行时的原子性以及与redis的事务的区别

在 Redis 中,Lua 脚本具有原子性是因为 Redis 保证在执行脚本时,脚本中的所有操作都会被当作一个不可分割的整体。具体来说,Redis 使用单线程的执行模型来处理命令,因此当 Lua 脚本在 Redis 中执行时,不会有其他命令打断脚本的执行过程。脚本中的所有操作都将连续执行,直到脚本执行完成后,Redis 才会继续处理其他客户端的请求。 Lua 脚本在 Redis 中原子性的原因

MySQL中一致性非锁定读

一致性非锁定读(consistent nonlocking read)是指InnoDB存储引擎通过多版本控制(multi versionning)的方式来读取当前执行时间数据库中行的数据,如果读取的行正在执行DELETE或UPDATE操作,这是读取操作不会因此等待行上锁的释放。相反的,InnoDB会去读取行的一个快照数据 上面展示了InnoDB存储引擎一致性的非锁定读。之所以称为非锁定读,因

InnoDB的多版本一致性读的实现

InnoDB是支持MVCC多版本一致性读的,因此和其他实现了MVCC的系统如Oracle,PostgreSQL一样,读不会阻塞写,写也不会阻塞读。虽然同样是MVCC,各家的实现是不太一样的。Oracle通过在block头部的事务列表,和记录中的锁标志位,加上回滚段,个人认为实现上是最优雅的方式。 而PostgreSQL则更是将多个版本的数据都放在表中,而没有单独的回滚段,导致的一个结果是回滚非

PHP: 深入了解一致性哈希

前言 随着memcache、redis以及其它一些内存K/V数据库的流行,一致性哈希也越来越被开发者所了解。因为这些内存K/V数据库大多不提供分布式支持(本文以redis为例),所以如果要提供多台redis server来提供服务的话,就需要解决如何将数据分散到redis server,并且在增减redis server时如何最大化的不令数据重新分布,这将是本文讨论的范畴。 取模算法 取模运

laravel框架实现redis分布式集群原理

在app/config/database.php中配置如下: 'redis' => array('cluster' => true,'default' => array('host' => '172.21.107.247','port' => 6379,),'redis1' => array('host' => '172.21.107.248','port' => 6379,),) 其中cl

Redis的rehash机制

在Redis中,键值对(Key-Value Pair)存储方式是由字典(Dict)保存的,而字典底层是通过哈希表来实现的。通过哈希表中的节点保存字典中的键值对。我们知道当HashMap中由于Hash冲突(负载因子)超过某个阈值时,出于链表性能的考虑,会进行Resize的操作。Redis也一样。 在redis的具体实现中,使用了一种叫做渐进式哈希(rehashing)的机制来提高字典的缩放效率,避

【吊打面试官系列-Redis面试题】说说 Redis 哈希槽的概念?

大家好,我是锋哥。今天分享关于 【说说 Redis 哈希槽的概念?】面试题,希望对大家有帮助; 说说 Redis 哈希槽的概念? Redis 集群没有使用一致性 hash,而是引入了哈希槽的概念,Redis 集群有 16384 个哈希槽,每个 key 通过 CRC16 校验后对 16384 取模来决定放置哪个槽, 集群的每个节点负责一部分 hash 槽。

Redis地理数据类型GEO

通常要计算两个地理位置的距离不是很方便,这里可以直接通过Redis提供的GEO操作来完成地理位置相关的计算 1)添加地理位置 语法:geoadd key longitude latitude member [longitude latitude member] ...字段说明:key:存放地理位置的集合名称longitude:地理坐标的经度latitude:地理坐标的纬度member:表示这