Redis 三大问题:缓存穿透、缓存击穿、缓存雪崩

embedded/2025/1/11 12:16:27/

Redis 作为高性能的内存数据库,广泛应用于缓存场景。然而,在实际使用中,可能会遇到三大经典问题:缓存穿透缓存击穿 和 缓存雪崩。这些问题如果不加以解决,可能会导致系统性能下降甚至崩溃。


1. 缓存穿透

问题描述

缓存穿透是指查询一个不存在的数据,由于缓存中没有该数据,请求会直接落到数据库上。如果大量这样的请求同时发生,数据库可能会被压垮。

原因

  • 请求的数据在缓存数据库中都不存在。
  • 恶意攻击者故意构造大量不存在的 key 进行请求。

解决方案

  1. 布隆过滤器(Bloom Filter)

    • 缓存层之前加一层布隆过滤器,用于快速判断数据是否存在。
    • 如果布隆过滤器判断数据不存在,则直接返回,避免查询数据库
  2. 缓存空值

    • 如果查询的数据不存在,将空值(如 null缓存起来,并设置较短的过期时间。
    • 后续请求可以直接从缓存中获取空值,避免频繁查询数据库
graph TDA[客户端请求] --> B{缓存中是否存在?}B -- 存在 --> C[返回缓存数据]B -- 不存在 --> D{布隆过滤器是否存在?}D -- 不存在 --> E[返回空值]D -- 存在 --> F[查询数据库]F -- 数据库中存在 --> G[返回数据并写入缓存]F -- 数据库中不存在 --> H[缓存空值并返回空值]

2. 缓存击穿

问题描述

缓存击穿是指某个热点 key 在缓存中过期,同时有大量请求访问该 key,导致请求直接落到数据库上,造成数据库压力骤增。

原因

  • 热点 key 在缓存中过期。
  • 大量请求同时访问该 key。

解决方案

  1. 设置热点 key 永不过期

    • 对于热点 key,可以设置永不过期,避免缓存失效。
    • 通过后台任务定期更新缓存
  2. 互斥锁(Mutex Lock)

    • 缓存失效时,使用分布式锁(如 Redis 的 SETNX)确保只有一个线程去查询数据库并更新缓存
    • 其他线程等待缓存更新完成后,直接从缓存中获取数据。

graph TDA[客户端请求] --> B{缓存中是否存在?}B -- 存在 --> C[返回缓存数据]B -- 不存在 --> D{获取分布式锁}D -- 获取成功 --> E[查询数据库]E -- 数据库中存在 --> F[写入缓存并返回数据]E -- 数据库中不存在 --> G[缓存空值并返回空值]D -- 获取失败 --> H[等待并重试]

3. 缓存雪崩

问题描述

缓存雪崩是指大量缓存 key 在同一时间失效,导致大量请求直接落到数据库上,造成数据库压力骤增甚至崩溃。

原因

  • 大量缓存 key 设置了相同的过期时间。
  • 缓存服务器宕机。

解决方案

  1. 设置随机过期时间

    • 缓存 key 设置随机的过期时间,避免大量 key 同时失效。
  2. 缓存高可用

    • 使用 Redis 集群或主从复制,确保缓存服务的高可用性。
  3. 限流降级

    • 缓存失效时,使用限流策略(如令牌桶算法)限制请求量,避免数据库被压垮。

graph TDA[客户端请求] --> B{缓存中是否存在?}B -- 存在 --> C[返回缓存数据]B -- 不存在 --> D{是否达到限流阈值?}D -- 未达到 --> E[查询数据库]E -- 数据库中存在 --> F[写入缓存并返回数据]E -- 数据库中不存在 --> G[缓存空值并返回空值]D -- 达到 --> H[返回降级数据或错误信息]

总结

Redis 的三大问题(缓存穿透、缓存击穿、缓存雪崩)是实际开发中常见的挑战。通过合理的设计和优化,可以有效避免这些问题:

  • 缓存穿透:使用布隆过滤器或缓存空值。
  • 缓存击穿:设置热点 key 永不过期或使用互斥锁。
  • 缓存雪崩:设置随机过期时间、缓存高可用和限流降级。


http://www.ppmy.cn/embedded/153000.html

相关文章

利用Java爬取1688商品详情API接口:技术与应用指南

引言 1688作为中国领先的B2B电子商务平台,拥有海量的商品信息。对于商家和市场研究人员来说,能够从1688获取商品详情信息,对于市场分析、竞品研究等具有重要价值。本文将详细介绍如何使用Java编写爬虫程序,以合法、高效的方式获取…

Mac Mini 最优雅的备份方式:使用极空间实现自动整机备份

Mac Mini 最优雅的备份方式:使用极空间实现自动整机备份 哈喽小伙伴们好,我是Stark-C~ 目前Mac mini M4到手已经一个多月,处理器性能大幅提升,内存也是破天荒的翻倍升级,再加上国补之后3500左右的价格,真…

Redis是单线程还是多线程?

大家好,我是锋哥。今天分享关于【Redis是单线程还是多线程?】面试题。希望对大家有帮助; Redis是单线程还是多线程? 1000道 互联网大厂Java工程师 精选面试题-Java资源分享网 Redis是 单线程 的。 尽管Redis的处理是单线程的&a…

(四)结合代码初步理解帧缓存(Frame Buffer)概念

帧缓存(Framebuffer)是图形渲染管线中的一个非常重要的概念,它用于存储渲染过程中产生的像素数据,并最终输出到显示器上。简单来说,帧缓存就是计算机图形中的“临时画布”,它储存渲染操作生成的图像数据&am…

SQL开窗函数相关的面试题和答案

基本排序与分组问题 题目:有学生成绩表tb_score,包含字段SNO(学号)、SCLASS(班级)、CHINESE(语文成绩)、ENGLISH(英语成绩)、ARITH(数学成绩&…

基于微信小程序的水果销售系统的设计与实现springboot+论文源码调试讲解

第4章 系统设计 一个成功设计的系统在内容上必定是丰富的,在系统外观或系统功能上必定是对用户友好的。所以为了提升系统的价值,吸引更多的访问者访问系统,以及让来访用户可以花费更多时间停留在系统上,则表明该系统设计得比较专…

Unity3D Huatuo热更环境安装与示例项目详解

前言 Unity3D作为一款强大的游戏开发引擎,广泛应用于各类游戏和应用程序的开发中。然而,随着游戏版本的迭代和功能的增加,热更新技术变得越来越重要。Huatuo是一款基于Unity3D的IL2CPP解释执行框架,可以实现对游戏代码的热更新&a…

计算机网络之---子网划分与IP地址

子网划分与IP地址的关系 在计算机网络中,子网划分(Subnetworking)是将一个网络划分为多个子网络的过程。通过子网划分,可以有效地管理和利用IP地址空间,提高网络的性能、安全性和管理效率。 子网划分的基本目的是通过…