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

网站前台用java做软件开发培训班

网站前台用java做,软件开发培训班,用node做的网站,网络营销实训总结报告title: redis读写一致问题 date: 2025-05-18 11:11:31 tags: redis categories: redis的问题方案 Redis读写一致问题 条件: 数据库此时的数据为10,redis此时的数据也为10 业务流程: 操作数据库使得数据库的数据为20,删除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.dtcms.com/wzjs/135570.html

相关文章:

  • 怎样做网站导购教程北京seo排名厂家
  • 云南手机网站开发信息推广服务
  • 可信网站认证代理班级优化大师客服电话
  • 网站建设费用计入什么二级科目网址百度刷排名
  • 武汉网站建设电话多少网站建站开发
  • 怎样申请一个免费网站网络服务主要包括
  • wordpress建图片站下载班级优化大师
  • 可以做盗版漫画网站吗互联网域名交易中心
  • 手机网站怎么开发超级优化大师下载
  • 广州品牌网站设计安徽疫情最新情况
  • 简答题网站建设步骤免费营销软件网站
  • 亚马逊网站建设历程友情链接交易网
  • 移动网站建设生要女宁德市教育局
  • 网站网页建设网络营销的真实案例分析
  • 网站建设公司业务培训上海高端网站定制
  • 秦皇岛建网站多少钱黄页网络的推广网站有哪些软件
  • 推广型网站如何建站微博推广价格表
  • 淄博网站建设相关文章西seo优化排名
  • 做外包胡it网站百度免费收录提交入口
  • php .net做网站哪个好今天热搜前十名
  • 建立网站做淘客百度网首页官网登录
  • 成熟网站开发单位石家庄seo优化
  • 做暧暧网站在线看seo快速排名服务
  • 广东深圳职业技术学院云巅seo
  • 网站建设一般步骤如何推广一个项目
  • 易语言跳到指定网站怎么做域名备案查询系统
  • 如何攻击织梦做的网站在线服务器网站
  • 网站如何改版云南网络营销公司
  • 建站快车代理商今天发生的重大新闻5条
  • 网站托管 济南360信息流广告平台