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

锦州网站建设公司这样做自己的网站

锦州网站建设公司,这样做自己的网站,中小企业融资平台,wordpress文章目录树【redis】redis缓存与数据库的一致性【1】四种同步策略【2】更新缓存还是删除缓存(1)更新缓存(2)删除缓存【3】先更新数据库还是先删除缓存(1)出现失败时候的情况1-先删除缓存,再更新数据库&…

【redis】redis缓存与数据库的一致性

  • 【1】四种同步策略
  • 【2】更新缓存还是删除缓存
    • (1)更新缓存
    • (2)删除缓存
  • 【3】先更新数据库还是先删除缓存
    • (1)出现失败时候的情况
      • 1-先删除缓存,再更新数据库(更新数据库失败了)
      • 2-先更新数据库,再删除缓存(删除缓存失败了)
      • 3-总结
    • (2)没有出现失败时候的情况
      • 1-先删除缓存,再更新数据库
        • 延时双删解决这个问题
        • 如果是读写分离的架构怎么办(强制读主库)
        • 删除失败了怎么办?
      • 2-先更新数据库,再删除缓存(最优方案)
        • 利用消息队列进行删除的补偿
  • 【4】总结

【1】四种同步策略

想要保证缓存与数据库的双写一致,一共有4种方式,即4种同步策略:
(1)先更新缓存,再更新数据库;
(2)先更新数据库,再更新缓存;
(3)先删除缓存,再更新数据库;
(4)先更新数据库,再删除缓存。

更新缓存与删除缓存哪种方式更合适?应该先操作数据库还是先操作缓存?

【2】更新缓存还是删除缓存

(1)更新缓存

(1)优点
每次数据变化都及时更新缓存,所以查询时不容易出现未命中的情况。

(2)缺点
更新缓存的消耗比较大。如果数据需要经过复杂的计算再写入缓存,那么频繁的更新缓存,就会影响服务器的性能。如果是写入数据频繁的业务场景,那么可能频繁的更新缓存时,却没有业务读取该数据。

(2)删除缓存

(1)优点
操作简单,无论更新操作是否复杂,都是将缓存中的数据直接删除。

(2)缺点
删除缓存后,下一次查询缓存会出现未命中,这时需要重新读取一次数据库。从上面的比较来看,一般情况下,删除缓存是更优的方案。

【3】先更新数据库还是先删除缓存

(1)出现失败时候的情况

首先,我们将先删除缓存与先更新数据库,在出现失败时进行一个对比:

1-先删除缓存,再更新数据库(更新数据库失败了)

先删除缓存再更新数据库,在出现失败时可能出现的问题:
(1)线程A删除缓存成功,线程A更新数据库失败;
(2)线程B从缓存中读取数据;由于缓存被删,进程B无法从缓存中得到数据,进而从数据库读取数据;此时数据库中的数据更新失败,线程B从数据库成功获取旧的数据,然后将数据更新到了缓存。
(3)最终,缓存和数据库的数据是一致的,但仍然是旧的数据

在这里插入图片描述

2-先更新数据库,再删除缓存(删除缓存失败了)

先更新数据库再删除缓存,在出现失败时可能出现的问题:
(1)线程A更新数据库成功,线程A删除缓存失败;
(2)线程B读取缓存成功,由于缓存删除失败,所以线程B读取到的是缓存中旧的数据。
(3)最后线程A删除缓存成功,有别的线程访问缓存同样的数据,与数据库中的数据是一样。
(4)最终,缓存和数据库的数据是一致的,但是会有一些线程读到旧的数据。

在这里插入图片描述

3-总结

经过上面的比较,我们发现在出现失败的时候,是无法明确分辨出先删缓存和先更新数据库哪个方式更好,以为它们都存在问题。上述场景出现的问题,应该如何解决呢?都建议采用重试机制解决。

(2)没有出现失败时候的情况

1-先删除缓存,再更新数据库

(1)线程A删除缓存成功;
(2)线程B读取缓存失败;
(3)线程B读取数据库成功,得到旧的数据;
(4)线程B将旧的数据成功地更新到了缓存;
(5)线程A将新的数据成功地更新到数据库。
在这里插入图片描述可见,进程A的两步操作均成功,但由于存在并发,在这两步之间,进程B访问了缓存。最终结果是,缓存中存储了旧的数据,而数据库中存储了新的数据,二者数据不一致。

延时双删解决这个问题

如果是先删缓存、再更新数据库,在没有出现失败时可能会导致数据的不一致。如果在实际的应用中,出于某些考虑我们需要选择这种方式,可以采用延时双删的策略,延时双删的基本思路如下:
(1)删除缓存;
(2)更新数据库;
(3)sleep N毫秒;
(4)再次删除缓存。

public void write(String key, Object data) {Redis.delKey(key);db.updateData(data);Thread.sleep(1000);Redis.delKey(key);
}

阻塞一段时间之后,再次删除缓存,就可以把这个过程中缓存中不一致的数据删除掉。而具体的时间,要评估你这项业务的大致时间,按照这个时间来设定即可。最终保证了数据库和缓存的数据一致

如果是读写分离的架构怎么办(强制读主库)

如果数据库采用的是读写分离的架构,那么又会出现新的问题,如下图:
此时来了两个请求,请求 A(更新操作) 和请求 B(查询操作)
(1)请求 A 更新操作,删除了 Redis;
(2)请求主库进⾏更新操作,主库与从库进行同步数据的操作;
(3)请 B 查询操作,发现 Redis 中没有数据;
(4)去从库中拿去数据;
(5)此时主从同步数据还未完成,拿到的数据是旧数据;

在这里插入图片描述
此时的解决办法就是如果是对 Redis 进行填充数据的查询数据库操作,那么就强制将其指向主库进⾏查询。

删除失败了怎么办?

如果删除依然失败,则可以增加重试的次数,但是这个次数要有限制,当超出一定的次数时,要采取报错、记日志、发邮件提醒等措施。

2-先更新数据库,再删除缓存(最优方案)

(1)线程A更新数据库成功;
(2)线程B读取缓存成功;
(3)线程A删除缓存成功。
在这里插入图片描述
可见,最终缓存与数据库的数据是一致的,并且都是最新的数据。但线程B在这个过程里读到了旧的数据,可能还有其他线程也像线程B一样,在这两步之间读到了缓存中旧的数据,但因为这两步的执行速度会比较快,所以影响不大。对于这两步之后,其他进程再读取缓存数据的时候,就不会出现类似于进程B的问题了。

利用消息队列进行删除的补偿

先更新数据库,后删除缓存这⼀种情况也会出现问题,比如更新数据库成功了,但是在删除缓存的阶段出错了没有删除成功,那么此时再读取缓存的时候每次都是错误的数据了。

此时解决方案就是利用消息队列进行删除的补偿。具体的业务逻辑⽤语⾔描述如下:
(1)请求 线程A 先对数据库进行更新操作;
(2)在对 Redis 进行删除操作的时候发现报错,删除失败;
(3)此时将Redis 的 key 作为消息体发送到消息队列中;
(4)系统接收到消息队列发送的消息后再次对 Redis 进行删除操作;

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

【4】总结

一般情况下,删除缓存是比更新缓存更优的方案;先更新数据库是比先删除缓存更优的方案;总的来说【先更新数据库,再删除缓存】就是四个策略中影响最小,效果最优的方案。

但是如果需要使用【先删除缓存,再更新数据库】的方案的话,可以使用【延时双删】【读写分离时强制读主库】【重试机制】来解决问题。

如果使用【先更新数据库,再删除缓存】时出现删除缓存失败的情况,可以使用【binlog同步到redis】来解决问题。

http://www.yayakq.cn/news/702023/

相关文章:

  • 安徽seo网站推广怎样做古玩网站
  • 深圳市住房和建设网站怎么优化网站源代码
  • 手机建站程序免费下载赣州优化
  • 哈尔滨开发网站网站创建想法
  • 山东手机响应式网站建设设计php可以做网站app吗
  • 哪个浏览器不屏蔽网站手机app推广联盟
  • 温州做网站费用软件定制开发方案模板
  • 什么叫网站收录导购返利网站开发
  • 最威海的网站建设服装企业网站模版
  • 大型网站 开发流程网站海外推广技巧
  • 息壤空间怎么上传网站滨州网站建设制作系统
  • 什么网站做热能表好网络营销论文怎么写
  • 湖南3合1网站建设公司网络推广引流是做什么的
  • 扶余网站建设营销软件app
  • 奕腾网站建设平顶山哪里做网站
  • 重庆手机网站推广方法网站 空间费用
  • 将二级域名 网站目录如何提高网站搜索排名
  • 网站专题页做多大尺寸知道ip域名如何进入网站
  • 做网站域名费向哪里交net源码的网站建设步骤
  • 白山市城乡建设局网站苏州装修公司网站建设
  • 顺德网站优化网站如何做修改密码的相关验证
  • 网站询盘量外贸商老贾的微博
  • 网站开发的母的目的和意义.wordpress制作教程视频
  • 广州城乡建设网站中国商标注册查询官网入口
  • 衡阳市住建局官方网站检察机关门户网站建设情况
  • wordpress 100万数据库营销型网站建设优化
  • 前端视频教程网站信息技术网站建设专业
  • 学做ppt的网站 免费下载织梦旅游网站模板
  • 网站做的好看的公司官网制作报价
  • 建设网站费用入会计分录外贸管理软件