MySQL与Redis如何保证数据一致性详解_数据库其它

来源:脚本之家  责任编辑:小易  

前言

由于缓存的高并发和高性能已经在各种项目中被广泛使用,在读取缓存这方面基本都是一致的,大概都是按照下图的流程进行操作:

但是在更新缓存方面,是更新完数据库再更新缓存还是直接删除缓存呢?又或者是先删除缓存再更新数据库?在这一点上就值得探讨了。

一致性方案

在实际项目开发中需要保证数据库和缓存中的数据一致,否则人家充值了100块,不断刷新却还是显示0.01元,岂不是尴尬?从理论上来说,为缓存设置过期时间是最终保证数据一致性的解决方案,采用这种方案的话,所有的写操作都是以数据库为准,如果数据库写入成功但是缓存更新失败,只要缓存到过期时间之后后面读缓存时自然会在数据库中读取新的值然后更新缓存。接下来探讨的思路主要的方向是在不依赖为缓存设置过期时间的前提下如何保证数据一致性。这里主要探讨三种方案:

①先更新数据库,再更新缓存
②先删除缓存,再更新数据库
③先更新数据库,再删除缓存

先更新数据库再更新缓存

这种方案是普遍被反对的(在我的认知范围中~),为啥呢?为啥这种方案就被反对呢?原因主要有两方面,请听我细细道来:

首先从数据安全方面考虑,如果同时有请求A和请求B同时进行操作,A先更新了数据库的一条数据,随后B马上有更新了该条数据,但是可能因为网络延迟等原因,B却比A先更新了缓存,就会出现一种什么情况呢?缓存中的数据并不最新的B更新过的数据,就导致了数据不一致的情况。

其次从业务场景方面考虑,如果是一个写数据库较多而读数据库较少的业务,如果采用这种方案就会导致数据还没读缓存就会被频繁更新,白白浪费性能。

综合以上两方面的考虑,这种方案果断pass。下面的两种方案就是争议较大的两种方案了,到底是先删缓存再更新数据库还是先更新数据库再删除缓存?

先删缓存再更新数据库

如果同时有一个请求A进行更新操作,请求B进行查询操作,就可能会出现A请求进行写操作前会删除缓存,B请求刚好此时进来发现缓存是空的,B请求就会查询数据库,如果此时A请求的写操作还未完成,B请求查询到的就还是旧的值,还是会将旧的值写入缓存,A请求将新的值写入数据库,此时就会导致数据不一致的问题,如果不采用给缓存设置过期时间的策略,该数据永远都是脏数据。

解决这种情况可以采用延时双删的策略,就是在更新数据库之前先删除缓存,然后对数据库进行写入操作,数据库更新完成之后再次进行删除缓存的操作,目的是删除读请求可能造成的缓存脏数据,第二次删除缓存之前可以休眠几秒,具体时间开发者可以评估一下自己项目读数据业务逻辑的耗时,然后在该耗时基础上加几百ms即可,这么做的目的就是确保读请求结束写请求可以删除读请求造成的脏数据。如果MySQL采用的是读写分离的架构,可能由于主从延时的原因造成数据不一致,可以在写操作完成之后根据主从延时时间休眠一下然后再进行删除缓存的操作。延时双删的伪代码如下:

# 伪代码
def delay_delete():
    redis.delete('name')  # 更新数据库之前先删除缓存
    sql = 'update info set name='lili' where id=1;'  # 更新数据库
    cursor.execute(sql)  
    time.sleep(1)  # 如果mysql是主从架构则休眠主从延时的时间再多几百ms
    redis.delete('name')  # 再次删除缓存

那会不会存在第二次删除缓存失败的情况呢?如果第二次删除失败,还是会造成缓存和数据库不一致的问题,又如何解决呢?且看下一种方案。

先更新数据库再删除缓存

老外提出了一个缓存更新方案Cache−AsidepatternCache-Aside patternCache−Asidepattern,文章中提到**应用程序应该从cache中获取数据,如果获取成功直接返回,如果没有获取成功,则从数据库中获取,成功后放到缓存中,更新数据时应该先把数据存到数据库中成功后再让缓存失效。**原文如下

If an application updates information, it can follow the write-through strategy by making the modification to the data store, and by invalidating the corresponding item in the cache.
When the item is next required, using the cache-aside strategy will cause the updated data to be retrieved from the data store and added back into the cache.

这种方案会不会产生数据不一致的情况呢?比如下述这种情况:

有两个请求A和B,A进行查询同时B进行更新,假设发生下述情况:

①此时缓存刚好失效

②请求A 就会去查询数据库得到一个旧的值

③请求B将新的值写入数据库

④请求B写入成功后删除缓存

⑤请求A将查到的机制写入缓存,产生脏数据...

如果发声上述情况,确实会产生数据不一致的情况,但是XDM想一想,发生这种情况的概率是多少呢?如果先要产生这种结果,就必须有一个条件,就是请求B的操作时间非常短,短到什么程度呢,就是请求B写入数据库的操作要比请求A从数据库中读取数据的速度要快(因为redis非常快,因此操作redis的时间可以暂且忽略),只有这种情况下④才可能比⑤先发声,但是数据库的读操作要远比写操作快的多,不然做读写分离干嘛呢?所以这种情况发生的概率是非常非常非常的低,但是如果强迫症患者出现必须要解决怎么办呢?就可以采用给缓存设置过期时间或者采用第二种方案的延时双删策略,保证读请求完成之后在进行删除操作。

最后的问题

还有问题呀,就是最终解决方案三可能 出现的极低概率的数据不一致的方案是采用方案二的延时双删策略,可是在方案二中也说了,如果出现缓存删除失败的情况咋办?那不是还会出现数据不一致的问题吗?这个问题到底如何解决呢?这里提供一个重试机制,删除失败就重试一次呗,这里提供一种重试的方案。

①更新数据库
②由于各种原因缓存删除失败
③将删除失败的缓存放入消息队列中
④业务代码从消息队列中获取需要删除的key
⑤继续尝试删除操作,直到成功

总结

到此这篇关于MySQL与Redis如何保证数据一致性的文章就介绍到这了,更多相关MySQL与Redis数据一致性内容请搜索真格学网以前的文章或继续浏览下面的相关文章希望大家以后多多支持真格学网!

您可能感兴趣的文章:聊一聊Redis与MySQL双写一致性如何保证详解redis缓存与数据库一致性问题解决mysql事务select for update及数据的一致性处理讲解如何恢复MySQL主从数据一致性分析Mysql事务和数据的一致性处理问题MySQL备份与恢复之保证数据一致性(5)mysql视图之确保视图的一致性(with check option)操作详解

  • 本文相关:
  • sql中代替like语句的另一种写法
  • 浅谈数据库缓存最终一致性的四种方案
  • win2003系统安装sql sever2000后1433端口未开放的解释
  • 经常使用的jdbc连接数据库方式
  • mac系统navicat的安装与使用教程
  • navicat如何导出所有的查询数据的方法
  • sqlserver中drop、truncate和delete语句的用法
  • 深入讲解sql中的字符串拼接
  • 介绍postgresql中的jsonb数据类型
  • postgres 数据库中的数据转换
  • 请教redis如何做到和mysql数据库的同步
  • mysql结合redis mysql断电 redis怎么办
  • 微博的系统架构,想用mysql+redis配合使用,想问一下具体要怎么...
  • mysql和redis如何同步java语言?
  • spring 怎么同时配置mysql和redis
  • redis mysql怎么配合
  • -redis中的数据怎样与mysql集群数据同步
  • php redis做mysql的缓存,怎么异步redis同步到mysql数据库?
  • redis怎么搭载mysql使用
  • 怎样劝服机关单位使用 MySQL/MongoDB/Redis 取代 Oracle
  • 网站首页网页制作脚本下载服务器操作系统网站运营平面设计媒体动画电脑基础硬件教程网络安全mssqlmysqlmariadboracledb2mssql2008mssql2005sqlitepostgresqlmongodbredisaccess数据库文摘数据库其它首页聊一聊redis与mysql双写一致性如何保证详解redis缓存与数据库一致性问题解决mysql事务select for update及数据的一致性处理讲解如何恢复mysql主从数据一致性分析mysql事务和数据的一致性处理问题mysql备份与恢复之保证数据一致性(5)mysql视图之确保视图的一致性(with check option)操作详解sql中代替like语句的另一种写法浅谈数据库缓存最终一致性的四种方案win2003系统安装sql sever2000后1433端口未开放的解释经常使用的jdbc连接数据库方式mac系统navicat的安装与使用教程navicat如何导出所有的查询数据的方法sqlserver中drop、truncate和delete语句的用法深入讲解sql中的字符串拼接介绍postgresql中的jsonb数据类型postgres 数据库中的数据转换navicat premium 15 永久破解激活navicat premium15安装及破解教程sql join on 用法sqlserver、mysql、oracle三种数mysql mysqldump命令使用详解sql中代替like语句的另一种写法如何让sql运行得更快关于adox的相关知识如何取得一个表的所有字段名用逗sql server下数据库链接的使用方在sql server中查询数据库中第几条至第几远程数据库的表超过20个索引的影响详细解在postgresql中实现递归查询的教程jdbc 数据库的连接(sqlserver oracle)特殊的存储过程-触发器概述navicat premium 12.0.29安装与激活超详细sqlite不支持right join的解决办法group access mysql mssql 随机 10条数据的sql语sybase 复制与热切换数据替换一个字段的所有非数字字符为空的sql语
    免责声明 - 关于我们 - 联系我们 - 广告联系 - 友情链接 - 帮助中心 - 频道导航
    Copyright © 2017 www.zgxue.com All Rights Reserved