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

redis读写一致问题


title: redis读写一致问题
date: 2025-05-18 11:11:31
tags: redis
categories: redis的问题方案

Redis读写一致问题

条件:

数据库此时的数据为10,redis此时的数据也为10

业务流程:

操作数据库使得数据库的数据为20,删除redis里面的数据保证读写一致

先删缓存,再操作数据库

出现读写不一致情况:

在这里插入图片描述

线程1(业务)线程2(并发线程)
删除缓存
查询缓存,没有命中,查询数据库(数据库查到为10,下一步将10写入redis)
将10写入缓存
更新数据库,将数据库中的数据改为20

最终情况

redis里面的数据数据库里面的数据
1020

出现数据不一致情况

先操作数据库,再删除缓存

在这里插入图片描述

线程1(并发线程)线程2(业务线程)
查询缓存未命中,查询数据库(下一步:将缓存更新为10)
更新数据库 v=20
删除缓存
写入缓存数据10

最终情况:

redis数据数据库数据
1020

两个方法选择原则

适用策略典型场景是否推荐使用延迟双删
先删缓存 → 后更新数据库高一致性业务(余额、库存)✅ 一定要延迟双删!
先更新数据库 → 后删缓存低一致性业务(资料、文章内容)❌ 可以不用延迟双删

解决方案:双写一致性

读操作没啥问题按照老流程

在这里插入图片描述

延时双删

在这里插入图片描述

问题答案
先删缓存还是先改数据库?先删缓存! 避免并发写入旧值
为什么删两次?防止“改库之后,又有人写了旧值到缓存”
为什么要延迟删?给并发线程一个“写入脏缓存”的机会,然后再清理掉它

缺点:

问题点延迟双删解决得了么?推荐改进方式
并发窗口写入脏缓存❌ 只能删最后一个分布式锁 + 双删 / MQ
延迟时间难控制❌ 不可预测MQ 或 Canal 机制更精准
异步删除失败风险❌ 会丢失删除使用可靠任务队列 / Redis 持久化
操作复杂、代码维护困难❌ 容易遗漏 key封装中间件、使用 AOP统一处理

给他加锁

读写都加锁

在这里插入图片描述

如图,程序运行串行化,性能低

引入共享锁和排他锁机制

共享锁:读锁readLock,加锁之后,其他线程可以共享读操作

排他锁:独占锁writeLock也叫,加锁之后,阻塞其他线程读写操作

在这里插入图片描述

代码Demo

import org.redisson.api.RReadWriteLock;
import org.redisson.api.RedissonClient;
import org.redisson.api.RLock;
import java.util.concurrent.TimeUnit;public class UserService {private final RedissonClient redissonClient;private final RedisService redisService;     // 你封装的 Redis 工具类private final UserRepository userRepository; // 你操作数据库的类public UserService(RedissonClient redissonClient, RedisService redisService, UserRepository userRepository) {this.redissonClient = redissonClient;this.redisService = redisService;this.userRepository = userRepository;}// 读操作:加“读锁”public User getUserById(Long userId) {String key = "user:" + userId;String lockKey = "lock:user:" + userId;RReadWriteLock rwLock = redissonClient.getReadWriteLock(lockKey);RLock readLock = rwLock.readLock();try {readLock.lock(5, TimeUnit.SECONDS); // 加读锁,防止同时写入User user = redisService.get(key);  // 先查缓存if (user != null) {return user;}// 缓存未命中 → 查数据库并回写缓存user = userRepository.findById(userId);if (user != null) {redisService.set(key, user, 10, TimeUnit.MINUTES);}return user;} finally {readLock.unlock(); // 释放读锁}}// 写操作:加“写锁”public void updateUser(User user) {Long userId = user.getId();String key = "user:" + userId;String lockKey = "lock:user:" + userId;RReadWriteLock rwLock = redissonClient.getReadWriteLock(lockKey);RLock writeLock = rwLock.writeLock();try {writeLock.lock(10, TimeUnit.SECONDS); // 加写锁,防止并发读/写redisService.del(key);                // 删除缓存(第一次)userRepository.save(user);            // 更新数据库// 第二次删除可延迟做(避免并发写入旧值)Thread.sleep(500);                    // 模拟延迟redisService.del(key);                // 延迟删除(第二次)} catch (InterruptedException e) {e.printStackTrace();} finally {writeLock.unlock(); // 释放写锁}}
}
import org.redisson.api.RReadWriteLock;
import org.redisson.api.RedissonClient;
import org.redisson.api.RLock;
import java.util.concurrent.TimeUnit;public class UserService {private final RedissonClient redissonClient;private final RedisService redisService;     // 你封装的 Redis 工具类private final UserRepository userRepository; // 你操作数据库的类public UserService(RedissonClient redissonClient, RedisService redisService, UserRepository userRepository) {this.redissonClient = redissonClient;this.redisService = redisService;this.userRepository = userRepository;}// 读操作:加“读锁”public User getUserById(Long userId) {String key = "user:" + userId;String lockKey = "lock:user:" + userId;RReadWriteLock rwLock = redissonClient.getReadWriteLock(lockKey);RLock readLock = rwLock.readLock();try {readLock.lock(5, TimeUnit.SECONDS); // 加读锁,防止同时写入User user = redisService.get(key);  // 先查缓存if (user != null) {return user;}// 缓存未命中 → 查数据库并回写缓存user = userRepository.findById(userId);if (user != null) {redisService.set(key, user, 10, TimeUnit.MINUTES);}return user;} finally {readLock.unlock(); // 释放读锁}}// 写操作:加“写锁”public void updateUser(User user) {Long userId = user.getId();String key = "user:" + userId;String lockKey = "lock:user:" + userId;RReadWriteLock rwLock = redissonClient.getReadWriteLock(lockKey);RLock writeLock = rwLock.writeLock();try {writeLock.lock(10, TimeUnit.SECONDS); // 加写锁,防止并发读/写redisService.del(key);                // 删除缓存(第一次)userRepository.save(user);            // 更新数据库// 第二次删除可延迟做(避免并发写入旧值)Thread.sleep(500);                    // 模拟延迟redisService.del(key);                // 延迟删除(第二次)} catch (InterruptedException e) {e.printStackTrace();} finally {writeLock.unlock(); // 释放写锁}}
}

中间件解决方案

异步通知保证数据的最终一致性

在这里插入图片描述

在这里插入图片描述

canal是基于mysql的主从同步来实现的

二进制日志(BINLOG)记录了所有的 DDL(数据定义语言)语句和 DML(数据操纵语言)语句,但不包括数据查询(SELECT、SHOW)语句。

http://www.xdnf.cn/news/514135.html

相关文章:

  • Visual Studio已更新为17.14+集成deepseek实现高效编程
  • AI大模型(二)embedding模型调用后对产生的数据进行分析
  • 水平可见直线--上凸包(andrew算法
  • 【嵙大o】C++作业合集
  • 不同版本 Linux 系统账号操作指令 ——rtkit 账号删除、普通账号的创建 / 删除 / 权限修改超详细大全
  • 如何在 Windows 11 或 10 上安装 Amazon Corretto
  • Ubuntu 20.04 报错记录: Matplotlib 无法使用 OpenCV 的 libqxcb.so
  • O2O电商变现:线上线下相互导流——基于定制开发开源AI智能名片S2B2C商城小程序的研究
  • Python蓝色飘雪
  • Linux云计算训练营笔记day10(MySQL数据库)
  • Java虚拟机 - JVM与Java体系结构
  • MyBatis 核心技术详解:从连接池到多表查询
  • Python多进程、多线程、协程典型示例解析
  • 深入理解 OpenCV 的 DNN 模块:从基础到实践
  • OpenSearch入门:从文档示例到查询实战
  • MCP - Cline 接入 高德地图 Server
  • DAY 29 复习日:类的装饰器
  • # 终端执行 java -jar example.jar 时(example.jar为项目jar包)报错:“没有主清单属性” 的解决方法
  • 第一章:重启之始
  • 零基础搭建!基于PP-ShiTuV2的轻量级图像识别系统(Docker+API部署指南)
  • 蓝桥杯1140 最小质因子之和(Hard Version)
  • 2KW压缩机驱动参考设计【SCH篇】
  • 使用conda创建python虚拟环境,并自定义路径
  • C++学习:六个月从基础到就业——C++20:协程(Coroutines)
  • Golang内存逃逸
  • 用代码解读_AI_强化学习在机器人路径规划中的应用与优化
  • nginx相关面试题30道
  • OpenCV-去噪效果和评估指标方法
  • MapReduce-WordCount实现按照value降序排序、字符小写、识别不同标点
  • 【ROS2】 核心概念6——通信接口语法(Interfaces)