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

[Redis] 终极缓存四连杀:缓存预热、缓存击穿、缓存穿透、缓存雪崩,真的懂了吗?

🎯 前言

你有没有遇到过这种情况:

  • 刚上线的新功能,所有用户一窝蜂冲进来,服务器被打爆?🚀(缓存预热)
  • 某个热点数据突然失效,数据库压力瞬间飙升,仿佛遭遇 DDoS?🔥(缓存击穿)
  • 有人恶意查询不存在的数据,导致数据库扛不住,全站崩溃?💀(缓存穿透)
  • 缓存整体崩溃,数据库直接被“屠杀”,网站直接GG?❄️(缓存雪崩)

这四大问题,是缓存系统的 噩梦级挑战,稍有不慎,就会导致网站宕机、业务瘫痪!

今天,我们就来揭秘这四个“终极杀手”,看看它们是什么,怎么解决,以及如何让你的缓存系统更加稳如老狗!🐶


一、缓存预热——让缓存从“0”到“1”

1. 什么是缓存预热?

缓存预热(Cache Warming) 就是 在系统启动或数据更新后,提前将常用数据加载到缓存,避免用户第一次访问时 直接打数据库,导致查询慢甚至数据库崩溃。

📌 打个比方: 想象你开了一家 网红奶茶店,门口排着 1000 个人!

  • 没做预热:第一个人来了,你才去煮珍珠、泡茶,结果一杯奶茶 10 分钟,用户等得要骂娘了。
  • 做了预热:提前煮好珍珠、泡好奶茶,用户一来就能迅速出杯,丝滑流畅!

高并发场景下,如果没有缓存预热,系统刚上线,所有请求都会打数据库,瞬间压垮数据库。

2. 解决方案

方案 1:启动时自动加载

  • 在 服务启动时,批量查询数据库的热门数据,放入缓存。
  • 适用于 数据变化不频繁,比如商品详情、系统配置。

方案 2:定时预热

  • 定时任务(CRON)每隔一段时间加载最新数据,适用于数据有一定变动的场景,比如 每日热销商品榜单。

方案 3:用户触发预热

  • 第一次查询数据库后,主动将数据写入缓存,适用于 数据更新不定时 的情况。

二、缓存击穿——“热点数据”消失的瞬间

1. 什么是缓存击穿?

缓存击穿(Cache Breakdown)指的是 某个热点 Key 在缓存中过期,短时间内大量请求打到数据库,导致数据库压力骤增甚至崩溃

感觉 Cache Breakdown 翻译为 缓存击穿并不好理解,理解为缓存瘫痪、崩溃反而更加贴近意思。

📌 打个比方: 你在天猫秒杀抢购一台 iPhone 15,结果查询 iphone_15_stock这个缓存 Key 失效了,所有用户的请求都去查询数据库,瞬间压垮 MySQL。

2. 解决方案

方案 1:设置热点数据不过期

  • 永不过期(TTL = -1),让热点数据一直在缓存中。
  • 适用于 热点数据稳定(如商品信息、排行榜)。

方案 2:使用“互斥锁”控制访问

  • 当缓存失效时,第一个查询的线程 加锁,查询数据库并更新缓存,其它线程等待:
if (redis.get("iphone_15_stock") == null) {
    if (redis.setnx("lock:iphone_15", 1)) { // 加锁
        int stock = db.query("SELECT stock FROM goods WHERE id=15");
        redis.set("iphone_15_stock", stock, 60); // 重新写缓存
        redis.del("lock:iphone_15"); // 释放锁
    } else {
        Thread.sleep(50); // 等待
        retry();
    }
}

方案 3:提前刷新缓存

  • 定时任务 在缓存快过期前 提前更新缓存,确保热点 Key 始终有效。

三、缓存穿透——数据库被“恶意”干爆

1. 什么是缓存穿透?

缓存穿透(Cache Penetration)指的是 查询一个根本不存在的数据,由于缓存没有该 Key,导致每次查询都直击数据库,造成数据库压力暴增!

📌 打个比方

  • 你去奶茶店点 唱跳rap特饮”,结果店员翻遍菜单都找不到,每次都要去后厨确认…… 数据库 = 被白嫖爆破!

在现实中,缓存穿透 通常由攻击者恶意构造请求,查询不存在的订单、用户 ID,导致数据库直接爆炸。

2. 解决方案

方案 1:缓存空对象

  • 如果数据库查询为空,仍然缓存一个空值,防止后续请求继续打数据库:
Object value = redis.get("user:99999");
if (value == null) {
    User user = db.query("SELECT * FROM user WHERE id=99999");
    if (user == null) {
        redis.set("user:99999", "", 60); // 缓存空对象 60 秒
    }
}

方案 2:布隆过滤器(Bloom Filter)

  • 使用布隆过滤器 维护一个 所有有效 Key 的集合,查询前先判断是否存在:
if (!bloomFilter.contains("user:99999")) {
    return null; // 直接返回,不查询数据库
}

方案 3:限流 + 黑名单

  • 对 IP 进行限流,防止某个 IP 短时间内大量请求 无效 Key,对恶意 IP 封禁处理。

方案 4:接口层增加校验

  • 用户鉴权、参数校验(请求参数是否合法、请求字段是否不存在等等);

四、缓存雪崩——比缓存击穿更致命

1. 什么是缓存雪崩?

缓存雪崩(Cache Avalanche)指的是 大量缓存 Key 在同一时间过期,导致请求全部打到数据库,数据库直接宕机。或者缓存服务出现故障,导致大量请求直接访问数据库,最终导致数据库崩溃的现象。

📌 打个比方

  • 你经营一个电商网站,给所有商品的缓存都设置了 同样的过期时间(比如 1小时)。
  • 结果,1 小时后 所有缓存同时失效,数据库瞬间被打爆,服务器直接跪了!

2. 解决方案

方案 1:给 Key 过期时间加随机数

  • 让不同 Key 的过期时间 错开,避免同一时间大面积失效:
redis.set("goods:123", data, 3600 + random(600)); // 1 小时 + 0~10 分钟随机时间

方案 2:分布式缓存架构

  • 使用 Redis Cluster,数据分散到多个节点,避免单点压力。

方案 3:双层缓存

  • 让 多个 Redis 共同承担缓存,防止单点崩溃。
  • 原理:使用两个缓存层,一个为主缓存(一级缓存),一个为从缓存(二级缓存)。数据首先存储在主缓存中,同时也在从缓存中有一份备份,但从缓存的过期时间设置得比主缓存长。当主缓存数据过期或者出现问题时,可以从从缓存中获取数据,为更新主缓存争取时间。
  • 示例:在一个金融交易系统中,对于股票价格信息,可以采用双缓存机制。主缓存中的股票价格数据过期时间较短,例如设置为 1 分钟,从缓存中的过期时间设置为 5 分钟。当主缓存中的股票价格数据过期时,系统可以先从从缓存中获取价格信息,同时更新主缓存。

🎯 总结

问题触发原因解决方案
缓存预热缓存刚上线,没有数据启动时加载、定时任务预热
缓存击穿热点 Key 失效,大量请求打数据库热点 Key 不过期、互斥锁、提前刷新
缓存穿透查询不存在的数据,导致数据库被打爆缓存空对象、布隆过滤器、限流黑名单
缓存雪崩大量缓存同时失效,数据库被压垮过期时间加随机值、分布式缓存、双层缓存

🚀 掌握这些缓存优化技巧,让你的 Redis 高并发系统更稳更快,拒绝崩溃!


http://www.kler.cn/a/572829.html

相关文章:

  • 为什么过滤器需要一个 Wrapper 来 extends HttpServletRequestWrapper
  • 神经网络为什么要用 ReLU 增加非线性?
  • DeepSeek R1模型医疗机构本地化部署评估分析(Discuss V1版上)
  • 探索IntelliJ IDEA的Tongyi Lingma插件:AI编程助手的新体验
  • Java项目中ES作为时序库
  • PHP之常量
  • 基于 Kubernetes 搭建 DevOps 持续集成环境
  • DeepSeek基础学习:深度讲解AI大模型基本原理
  • Unity:物体指定初速度、方向移动方法
  • day1 第二次入门rust
  • dify、open-webui、chatbox 对比
  • MFC扩展库BCGControlBar Pro v36.1新版亮点 - 对话框表单组件升级
  • Linux 命令终极指南:分类详解与实战场景
  • 内核编程八:基于printk宏的pr_* 宏
  • Pywinauto Recorder: 将Windows操作转化为Python脚本,高效简易地实现操作自动化
  • 【单片机通信技术】STM32 HAL库 SPI主从机通过串口发送数据
  • ASP.NET Core 3.1 修改个别API返回JSON序列化格式
  • e2studio开发RA2E1(15)----配置RTC时钟及显示时间
  • 如何排查服务器内存泄漏问题
  • Java 导出 PDF 文件:从入门到实战