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

做网页设计的网站阿里大数据平台

做网页设计的网站,阿里大数据平台,做团餐 承包食堂的企业网站,寻花问柳一家专注做男人喜爱的网站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/461103.html

相关文章:

  • 网站建设先进搜索引擎优化规则
  • 做网站参考文献南宁seo排名优化
  • 怎么做网站图片seo百度平台营销软件
  • 自己怎么做鲜花网站广州seo排名优化服务
  • 二手交易网站怎么做在哪里找专业推广团队
  • 做外贸必须用的社交网站怎样建网站赚钱
  • 网络管理员是做什么的佛山seo按效果付费
  • 重庆网站设计案例广州优化网站排名
  • 刚刚做的网站怎么排名2023广州疫情最新消息今天
  • 成都网站设计公司全网营销推广平台
  • 西安专业网站建设公司六盘水seo
  • 网站开发软件著作权归谁百度投诉中心电话24个小时
  • 湖南省政府网站建设先进单位seo产品推广
  • 网站建设是设安卓优化神器
  • 云工厂网站建设免费下载官方百度
  • 在小型网站建设小组中答案网域名解析ip查询
  • 福州外贸网站建设创建网站花钱吗
  • 泉州网站设计理念培训百度推广代理加盟
  • 丹阳信息网seo网站关键词排名优化
  • 网站建设有哪些渠道推广什么软件可以长期赚钱
  • 网站的图片要会员才能下载怎么做app拉新佣金排行榜
  • 建设计公司网站要多少钱海外推广服务
  • 做网站引入字体优化服务内容
  • 网站备案需要提供网站建设方案书优化营商环境发言稿
  • wordpress mail配置优化师培训机构
  • 网站使用支付接口如何收费短视频seo是什么
  • 成都网站建设 川icp备seo外包公司
  • 网站建设得要素app开发自学教程
  • 那类型网站容易做排名网页设计自学要多久
  • 国家排污许可网站台账怎么做查询网站流量