news 2026/6/15 13:05:58

Redis缓存三大问题详解:击穿、穿透与雪崩的解决方案

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
Redis缓存三大问题详解:击穿、穿透与雪崩的解决方案

在使用 Redis 作为缓存层时,我们经常会遇到三个经典问题:缓存击穿、缓存穿透和缓存雪崩。这些问题可能导致系统性能下降甚至崩溃,本文将详细介绍这三个问题的原因和解决方案。

一、🎯 缓存击穿

问题描述:某个热点 key 在缓存中过期的瞬间,大量请求同时访问数据库,造成数据库压力骤增。
解决方案:

1.1 设置热点永不过期

案例代码:

// 对于热点数据,可以不设置过期时间 redisTemplate.opsForValue().set("hot_data", data);

1.2 互斥锁机制

使用互斥锁防止多个线程同时访问数据库。

案例代码:

public String getDataWithMutex(String key) { String value = redisTemplate.opsForValue().get(key); if (value == null) { // 获取分布式锁 Boolean lock = redisTemplate.opsForValue() .setIfAbsent("lock:" + key, "1", Duration.ofSeconds(10)); if (lock) { try { // 查询数据库 value = queryFromDatabase(key); // 写入缓存 redisTemplate.opsForValue().set(key, value, Duration.ofMinutes(5)); } finally { // 释放锁 redisTemplate.delete("lock:" + key); } } else { // 等待其他线程加载完成 Thread.sleep(100); return getDataWithMutex(key); } } return value; }

1.3 接口限流与熔断

重要接口设置限流策略,防止恶意刷接口,同时进行熔断处理

二、🔍 缓存穿透

问题描述:查询一个不存在的数据,缓存和数据库都查不到,每次请求都会打到数据库,浪费系统资源。

解决方案:

2.1 接口层增加校验

如用户鉴权校验,id做基础校验,id<=0的直接拦截

2.2 缓存空值

对于缓存和数据库中都没有的数据,将key-value对写为key-null,设置短期有效时间

代码如下:

public String getDataWithNullCache(String key) { String value = redisTemplate.opsForValue().get(key); if (value == null) { // 查询数据库 value = queryFromDatabase(key); if (value == null) { // 缓存空值,设置较短过期时间 redisTemplate.opsForValue().set(key, "", Duration.ofMinutes(1)); return null; } else { // 缓存真实数据 redisTemplate.opsForValue().set(key, value, Duration.ofMinutes(10)); } } else if (value.isEmpty()) { // 缓存的空值 return null; } return value; }

2.3 布隆过滤器

使用布隆过滤器快速判断数据是否存在于缓存中,避免频繁访问数据库。

什么是布隆过滤器?

------------布隆过滤器(Bloom Filter)是1970年由布隆提出的。它实际上是一个很长的二进制向量和一系列随机映射函数。布隆过滤器可以用于检索一个元素是否在一个集合中。它的优点是空间效率和查询时间都比一般的算法要好的多,缺点是有一定的误识别率和删除困难。以Redis中的布隆过滤器实现为例,Redis中的布隆过滤器底层是一个大型位数组(二进制数组)+多个无偏hash函数。在布隆过滤器增加元素之前,首先需要初始化布隆过滤器的空间,也就是上面说的二进制数组,除此之外还需要计算无偏hash函数的个数。隆过滤器提供了两个参数,分别是预计加入元素的大小n,运行的错误率f。布隆过滤器中有算法根据这两个参数会计算出二进制数组的大小l,以及无偏hash函数的个数k。布隆过滤器支持增加和查询,不支持删除和修改。 关系就是:错误率越低,位数组越长,控件占用较大;错误率越低,无偏hash函数越多,计算耗时较长

案例代码:

@Component public class BloomFilterService { private final RedisTemplate<String, Object> redisTemplate; // 添加数据到布隆过滤器 public void addToBloomFilter(String key) { redisTemplate.opsForValue().setBit("bloom:user", hash(key), true); } // 检查数据是否存在 public boolean mightContain(String key) { long hash1 = hash(key); long hash2 = hash2(key); for (int i = 0; i < 3; i++) { long combinedHash = hash1 + i * hash2; if (!redisTemplate.opsForValue().getBit("bloom:user", combinedHash)) { return false; } } return true; } private long hash(String key) { return key.hashCode() & Integer.MAX_VALUE; } private long hash2(String key) { return (key.hashCode() >> 16) & Integer.MAX_VALUE; } }

布隆过滤器的其他典型应用

  • 网页URL去重:爬虫系统中判断网页是否已抓取
  • 垃圾邮件过滤:判断邮件地址是否为垃圾邮件发送者
  • 数据库查询优化:避免查询不存在的数据记录
  • 推荐系统:过滤用户已看过的内容

三、❄️ 缓存雪崩

问题描述:大量缓存 key 在同一时间过期,导致大量请求直接打到数据库,造成数据库瞬时压力过大。

解决方案:

3.1 过期时间随机化

避免大量数据同时过期

public void setCacheWithRandomExpire(String key, Object value) { // 基础过期时间 + 随机时间 int baseExpire = 300; // 5分钟 int randomExpire = new Random().nextInt(600); // 0-10分钟随机 redisTemplate.opsForValue().set( key, value, Duration.ofSeconds(baseExpire + randomExpire) ); }

3.2 构建高可靠集群

通过主从节点方式构建Redis缓存高可靠集群,避免单点故障。

3.3 服务降级与限流

在缓存雪崩发生时,启动请求限流机制,减少数据库压力

@Component public class CacheProtectionAspect { private final RateLimiter rateLimiter = RateLimiter.create(1000); // 限流1000QPS @Around("@annotation(com.example.annotation.CacheProtected)") public Object protectCacheAccess(ProceedingJoinPoint joinPoint) throws Throwable { if (!rateLimiter.tryAcquire()) { // 限流触发,返回默认值或抛出异常 throw new RuntimeException("系统繁忙,请稍后重试"); } try { return joinPoint.proceed(); } catch (Exception e) { // 降级处理 return getDefaultFallbackValue(); } } private Object getDefaultFallbackValue() { // 返回默认值或从备用数据源获取 return "default_value"; } }

总结缓存击穿、穿透和雪崩的区别可从下图看出:

redis缓存击穿、穿透、雪崩的了解就到这里了,这部分内容在软考中也可能会被考到。

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/6/12 13:29:22

05-矩阵理论复习第五章 向量与矩阵范数

本文内容为研究生矩阵理论复习第五章 向量与矩阵范数笔记记录&#xff0c;包含了向量的1范数&#xff0c;2范数&#xff0c;无穷范数&#xff0c;以及矩阵的1范数&#xff0c;2范数&#xff0c;无穷范数&#xff0c;与F范数等知识内容&#xff0c;还介绍了谱半径相关PDF文档可前…

作者头像 李华
网站建设 2026/6/13 22:04:32

金仓数据库PL/SQL兼容性深度评测:为什么说它最接近Oracle?

引言 在国产数据库替代Oracle的浪潮中,PL/SQL兼容性成为企业迁移的关键考量因素。作为Oracle数据库的核心编程语言,PL/SQL承载了大量业务逻辑。本文将深入对比金仓数据库(KingbaseES)与达梦数据库在PL/SQL开发体验上的差异,帮助开发者和架构师做出更明智的选择。 一、PL/SQL兼容…

作者头像 李华
网站建设 2026/6/12 14:42:05

一站式了解长轮询,SSE和WebSocket

文章目录引言1.Long Polling (长轮询)2. SSE (Server-Sent Events)3. WebSocket对比总结引言 作为后端开发人员&#xff08;Java/Go&#xff09;&#xff0c;你肯定遇到过需要实时将服务器数据推送到客户端的场景。传统的 HTTP 请求-响应模式无法满足这种需求。 为了解决这个…

作者头像 李华
网站建设 2026/6/4 21:01:24

货运管理系统:打造货运行业的数字化管理利器

温馨提示&#xff1a;文末有资源获取方式~ 一、货运搬家系统&#xff1a;让货运搬家更高效、更便捷 在当今快节奏的生活和商业环境中&#xff0c;货运搬家是人们和企业经常面临的需求。无论是个人搬家、企业搬迁&#xff0c;还是货物运输&#xff0c;都需要一个高效、便捷的…

作者头像 李华
网站建设 2026/6/10 18:08:53

RFID资产管理:数据安全措施全解析,企业必看指南

在数字化资产管理进程中&#xff0c;RFID 技术凭借高效识别、批量采集的特性&#xff0c;成为企业提升资产管理效率的重要手段&#xff0c;而数据安全则是 RFID 资产管理体系落地的核心根基。首码资产管理系统依托 RFID 技术搭建全流程数据防护体系&#xff0c;从数据采集到存储…

作者头像 李华