首先关于两者数据的一致性包含有两种情况:

 (1)缓存中有数据时,那数据库中的数据要和缓存中的数据相同;

 (2)缓存中没有数据时,数据库中的数据必须是最新的。

如果不符合以上两种情况,就属于缓存和数据库数据不一致的问题。

缓存不同的读写模式,所对应的不一致问题也会有所差异,缓存的读写模式主要有两种,读写缓存和只读缓存。

1.对于读写缓存来讲,需要关注缓存数据写完以后,对于数据库的写会策略,主要分为两种:

 (1)同步直写,写缓存的时候,也同步写数据库,这样数据一致,改策略适用于需要读写缓存和数据库数据一致的场景,但使用这种策略需要注意,缓存和数据库中的数据需要同时更新,所以需要在更新中使用事务机制,保证缓存和数据库的更新具有原子性,要么同时更新成功,要么都更新失败并返回错误信息,进行重试。

 (2)异步写回,写缓存时不写回数据库,等缓存中数据要被淘汰时,再写回数据库,但如果写完缓存,还未更新数据库时缓存出现故障,那要数据库中就没有最新的数据了,该种策略适用于对数据一致性要求不是那么高的场景,如电商商品的非关键属性和短视频的创建和修改时间等,可以使用该策略。

2.只读缓存的数据变更主要分为,增加新数据和删改已有的数据:

 (1)新增数据,对于新增数据,只需要直接添加到数据库中即可,不需要对缓存做什么操作。这样数据库中是最新的数据,缓存中没有新增数据,对应前面的数据一致性的第二种情况,也是数据一致的。

 (2)删改数据,如果发生删改数据,既要删改数据库中的数据,还要让缓存中的数据无效,同时还需要保证两个操作的原子性,要么都执行成功,要么都执行失败,不然都会导致数据不一致的情况发生,

    a.先更新数据库,后删缓存。可能发生缓存删除失败,下次查询命中缓存,但读取到的是旧值

    b.先删除缓存,后更新数据库。可能发生数据库更新失败,下次查询时缓存中没有,从数据库中获取的是未更新的旧值。

    解决该种场景下的数据不一致问题,可以采用重试方法,考虑使用如Kafka消息队列,把要删改的数据放入到Kafka中,只有缓存和数据库同时操作成功时,才将其从消息队列中删除,否则重新读取并操作,重试次数超过一定限制时返回相应的错误。

    对应上面两种删除的场景和顺序,由于执行先后的时间差的缘故,也有可能导致不同的数据不一致的问题出现,但也可以有针对性的不同解决策略:

      1. 针对先删除缓存再更新数据库的场景,可能会出现线程A先删除了缓存但并没来得急更新数据库,这时线程B来读取,缓存中没有,将从数据库中读取,a.但读到的是旧值,b.同时还会导致将该旧值同步到缓存中。

        解决方法:可以再线程A更新完数据库时,sleep一小段时间(这个时间要大于线程B从数据库中读取数据并更新缓存的时间),然后线程A再删除缓存,之后再有其他线程来读取时,发现缓存缺失,则会从数据库中读取最新的值,并更新至缓存中

             这个方法就叫做延迟双删。

      2. 针对先更新数据库,后删除缓存这种场景,可能存在线程A先更新了数据库还没来及删除缓存的时候,线程B来读取并命中缓存,但是读到了旧值

        解决方法:也可以使用延迟双删

    总结下来:(1)对于更新数据库或者删除缓存可能失败导致的不一致,可以采用重试的方式解决;

         (2)对于删除缓存和更新数据库过程中,由于存在其他并发线程读取而导致的不一致问题,可以采用延迟双删的方法来解决。

    建议使用先更新数据库,再删除缓存的顺序,原因如下:

        (1)先删缓存,再更新数据库,可能由于缓存的缺失而访问数据库,进而给数据库带来压力

        (2)如果业务应用中读取数据库和写缓存的时间不好估算,那么,延迟双删中的等待时间就不好设置。

PS:如果业务层要求必须读取一致性的数据,可以在更新数据库的同时,现在redis缓存暂存客户端并发读请求,等数据库更新完,缓存删除完,再来读取,保证数据一致性。

参考《Redis核心技术与实战》蒋德钧

Redis缓存中的数据和数据库不一致的更多相关文章

  1. springboot中如何向redis缓存中存入数据

    package com.hope;import com.fasterxml.jackson.core.JsonProcessingException;import com.fasterxml.jack ...

  2. [转]在nodejs使用Redis缓存和查询数据及Session持久化(Express)

    本文转自:https://blog.csdn.net/wellway/article/details/76176760 在之前的这篇文章 在ExpressJS(NodeJS)中设置二级域名跨域共享Co ...

  3. 基于Python项目的Redis缓存消耗内存数据简单分析(附详细操作步骤)

    目录 1 准备工作 2 具体实施   1 准备工作 什么是Redis? Redis:一个高性能的key-value数据库.支持数据的持久化,可以将内存中的数据保存在磁盘中,重启的时候可以再次加载进行使 ...

  4. EF封装类 增加版,增加从缓存中查找数据方法,供参考!

    EF封装类 增加版,增加从缓存中查找数据方法,供参考! 这个类是抽象类,我这里增加了需要子类验证的方法ValidateEntity,方便扩展,若想直接使用该类,可以将该类更改成静态类,里面所有的方法都 ...

  5. 在nodejs使用Redis缓存和查询数据及Session持久化(Express)

    在nodejs使用Redis缓存和查询数据及Session持久化(Express) https://segmentfault.com/a/1190000002488971

  6. 简单实现Redis缓存中的排序功能

    1.在实现缓存排序功能之前,必须先明白这一功能的合理性.不妨思考一下,既然可以在数据库中排序,为什么还要把排序功能放在缓存中实现呢?这里简单总结了两个原因:首先,排序会增加数据库的负载,难以支撑高并发 ...

  7. redis(四)--简单实现Redis缓存中的排序功能

    在实现缓存排序功能之前,必须先明白这一功能的合理性.不妨思考一下,既然可以在数据库中排序,为什么还要把排序功能放在缓存中实现呢?这里简单总结了两个原因:首先,排序会增加数据库的负载,难以支撑高并发的应 ...

  8. Redis缓存中的常见问题

    缓存穿透:是指查询一个Redis和数据库中都不存在的数据. 问题:查询一个Redis和数据库中都不存在的数据,大量请求去访问数据库,导致数据库宕机. 解决办法: 1.根据id查询,如果id是自增的,将 ...

  9. 获取apache ignite缓存中的数据行数少于实际行数

    我将ignite项目打包放到linux下,在linux下获取window中存放在oracle数据库中的数据,linux服务器作为ignite的服务端节点,我在本地启动tomact,作为ignite客户 ...

  10. Java -> 把Excel表格中的数据写入数据库与从数据库中读出到本地 (未完善)

    写入:没有关闭流,容错并不完善. private void insertFile(HttpServletRequest request, HttpServletResponse response) t ...

随机推荐

  1. IO相关了解

    1.IO的概念 IO简单来讲就是对输入输出设备的简化表达形式 单片机中各种接口,进行数据流的传输 从磁盘中读取数据至内存,又或者从内存中写入磁盘 编程中的IO 此时的IO其应用程序的运行态,即进程-- ...

  2. Word11 工程学院讲师论文office真题

    1.根据题目一的要求,打开素材文件,点击[文件]-[另存为],选择[当前文件夹],命名为Word. 2.根据题目二的要求,根据提供的参考样式,打开Word的文档,在[布局]里点击[页面设置]的右下角, ...

  3. Mysql数据库基础第五章:(二)视图

    Mysql数据库基础系列 软件下载地址 提取码:7v7u 数据下载地址 提取码:e6p9 mysql数据库基础第一章:(一)数据库基本概念 mysql数据库基础第一章:(二)mysql环境搭建 mys ...

  4. csp201503-2(数字排序)

    问题描述 给定n个整数,请统计出每个整数出现的次数,按出现次数从多到少的顺序输出. 输入格式 输入的第一行包含一个整数n,表示给定数字的个数. 第二行包含n个整数,相邻的整数之间用一个空格分隔,表示所 ...

  5. 前端常见的Vue面试题目汇总

    请说一下响应式数据的原理 默认Vue在初始化数据时,会给data中的属性使用Object.defineProperty重新定义所有属性,当页面到对应属性时,会进行依赖收集(收集当前组件中的watche ...

  6. 【随手记】Burp Suite 设置HTTP2

    Burp Suite 设置HTTP2

  7. docker 二进制安装

    首先所属环境为内网并且服务器拥有的开发环境不确定,需要跑当前服务所需代码,所以优先选择使用docker docker 文档地址 https://docs.docker.com 在 install 中存 ...

  8. 性能测试工具locust压测介绍

    官方文档:https://docs.locust.io/en/stable/index.html 1.初识locust Locust 完全基本 Python 编程语言,采用python 编写压测脚本, ...

  9. PHP安装SOAP扩展调用webservice获取数据

    报错内容: 调用方式: 错误原因: URL未加后缀?WSDL导致异常,加入后异常问题解决.

  10. 设置view的圆角和阴影

    1.设置view圆角 self.backView.clipsToBounds = YES; self.backView.layer.cornerRadius = 6.f; 2.设置view阴影 sel ...