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

Redis缓存实战:彻底解决缓存穿透、击穿、雪崩三大难题

一、缓存穿透

问题描述

查询一个数据库和缓存中都不存在的key,每次请求都打到数据库,大量请求可能拖垃数据库。

解决方案

1. 缓存空对象

@Service public class UserService { @Autowired private RedisTemplate<String, Object> redisTemplate; @Autowired private UserMapper userMapper; public User getUserById(Long id) { String key = "user:" + id; // 查询缓存 User user = (User) redisTemplate.opsForValue().get(key); if (user != null) { return user.getId() == null ? null : user; // 返回空对象表示不存在 } // 查询数据库 user = userMapper.selectById(id); if (user == null) { // 缓存空对象,设置短过期时间 redisTemplate.opsForValue().set(key, new User(), 5, TimeUnit.MINUTES); return null; } redisTemplate.opsForValue().set(key, user, 30, TimeUnit.MINUTES); return user; } }

2. 布隆过滤器:将所有合法key存入Bloom Filter,请求先过滤器,不存在的key直接拖到。

回到顶部

二、缓存击穿

问题描述

查询一个缓存刚好失效的key,大量请求同时打到数据库。

解决方案

互斥锁(推荐)

public User getUserById(Long id) { String key = "user:" + id; User user = (User) redisTemplate.opsForValue().get(key); if (user != null) return user; // 加分布式锁 String lockKey = "lock:user:" + id; Boolean locked = redisTemplate.opsForValue() .setIfAbsent(lockKey, "1", 10, TimeUnit.SECONDS); if (Boolean.TRUE.equals(locked)) { try { // 再次查询缓存(可能其他线程已经写入) user = (User) redisTemplate.opsForValue().get(key); if (user == null) { user = userMapper.selectById(id); redisTemplate.opsForValue().set(key, user, 30, TimeUnit.MINUTES); } } finally { redisTemplate.delete(lockKey); } } else { // 未获得锁,稍等重试 Thread.sleep(50); return getUserById(id); } return user; }

回到顶部

三、缓存雪崩

问题描述

大量缓存key同时失效,大量请求同时打到数据库。

解决方案

1. 过期时间加随机偶数

// 设置缓存时加随机偶数,避免同时失效 int randomExpire = 30 + new Random().nextInt(10); // 30~40分钟 redisTemplate.opsForValue().set(key, user, randomExpire, TimeUnit.MINUTES);

2. 多级缓存:本地缓存(Caffeine)+ Redis两级,即使 Redis雪崩也有本地缓存托底。

3. 缓存预热:项目启动时提前将热点数据加载到缓存。

回到顶部

总结

问题原因解决方案
缓存穿透key不存在空对象 / 布隆过滤器
缓存击穿key失效瞬间互斥锁 / 逆机制
缓存雪崩大量同时失效随机过期 / 多级缓存
http://www.gsyq.cn/news/1611599.html

相关文章:

  • 整群随机抽样
  • 2026 年本地部署的 AI 数字人直播系统到底有多省钱?
  • 从Prompt到Loop,拆解Agent 进化的底层逻辑
  • 北京心商科技发布GEO优化产品 助力企业合规营销
  • 开源模型技术总结-3————FireRed(小红书)开源模型
  • 主流三维光学轮廓仪国产品牌与技术解析
  • 】[SampleVirtualTexture节点]原理解析与实际应用
  • 操作简便吗?8款AI写作辅助网站综合榜,毕业答辩稳了!
  • 2026 在上海如何找一家专业又靠谱的小程序定制开发公司
  • 告别命令行!用JGit在Java项目里优雅地操作Git(附完整API使用示例)
  • 2026最新年版Java八股文,7天背完通过面试
  • 机器人电源方案评审清单:别只看电压电流,还要看这 6 件事
  • 2026年本地部署的AI数字人直播系统 Top5 实测
  • ArkUI 底部操作栏及卡片整体美化布局开发
  • 从“工作记忆”到“资源博弈”:AI Agent 的 Context Window 为何是最核心的工程约束?
  • 垃圾自动分类技术:从AI识别到机械分拣的工程实践与选型指南
  • 20260630 - 看门狗
  • Go语言面试遇到,面试官问什么是协程、什么是协程泄漏和数组跟切片是用该如何回答
  • Agent Skill 学习笔记
  • 基于.NET AgentFramework开发OpenClaw智能体框架
  • GESP2026年6月认证C++三级( 第三部分编程题(1、加密))精讲
  • 深入浅出理解卷积的概念
  • 告别锁竞争:用C++11的concurrentqueue重构你的生产者消费者模型(附完整代码)
  • 一天一个Python库:tomlkit - 轻松解析和操作TOML配置
  • Python基础:三元表达式极简写法与高阶嵌套、场景避坑指南
  • 基于51/STM32单片机分贝仪检测 噪音等级声音采集(设计源文件+万字报告+讲解)(支持资料、图片参考_降重降ai)
  • 终极指南:如何安全备份微信聊天记录的技术方案解析
  • AKShare:金融数据接口的架构哲学与实践反思
  • SPT-AKI Profile Editor:逃离塔科夫离线服务器存档修改终极指南
  • 【VMware网络专家20年压箱底笔记】:多虚拟机通信必须绕开的4个致命陷阱(第3个连vCenter日志都不报错)