[SpringBoot] 解决Redis相关问题
创始人
2025-05-28 07:12:04
0
  • 缓存穿透
  • 缓存击穿
  • 缓存雪崩

文章目录

      • 1.缓存穿透
      • 2.缓存击穿
      • 3.缓存雪崩

1.缓存穿透

缓存穿透指的是一个缓存系统无法缓存某个查询的数据,从而导致这个查询每一次都要访问数据库。

常见的Redis缓存穿透场景包括:

  1. 查询一个不存在的数据:攻击者可能会发送一些无效的查询来触发缓存穿透。
  2. 查询一些非常热门的数据:如果一个数据被访问的非常频繁,那么可能会导致缓存系统无法处理这些请求,从而造成缓存穿透。
  3. 查询一些异常数据:这种情况通常发生在数据服务出现故障或异常时,从而造成缓存系统无法访问相关数据,从而导致缓存穿透。

使用Guava在内存中维护一个布隆过滤器

com.google.guavaguava29.0-jre
org.springframework.bootspring-boot-starter-data-redis

缓存中维护Bloom Filter

public class BloomFilterUtil {// 布隆过滤器的预计容量private static final int expectedInsertions = 1000000;// 布隆过滤器误判率private static final double fpp = 0.001;private static BloomFilter bloomFilter = BloomFilter.create(Funnels.stringFunnel(Charset.defaultCharset()), expectedInsertions, fpp);/*** 向Bloom Filter中添加元素*/public static void add(String key){bloomFilter.put(key);}/*** 判断元素是否存在于Bloom Filter中*/public static boolean mightContain(String key){return bloomFilter.mightContain(key);}
}

在Controller中查询数据时,先根据请求参数进行Bloom Filter的过滤

@Autowired
private RedisTemplate redisTemplate;@GetMapping("/user/{id}")
public User getUserById(@PathVariable Long id){// 先从布隆过滤器中判断此id是否存在if(!BloomFilterUtil.mightContain(id.toString())){return null;}// 查询缓存数据String userKey = "user_"+id.toString();User user = (User) redisTemplate.opsForValue().get(userKey);if(user == null){// 查询数据库user = userRepository.findById(id).orElse(null);if(user != null){// 将查询到的数据加入缓存redisTemplate.opsForValue().set(userKey, user, 300, TimeUnit.SECONDS);}else{// 查询结果为空,将请求记录下来,并在布隆过滤器中添加BloomFilterUtil.add(id.toString());}}return user;
}

2.缓存击穿

缓存击穿指的是在一些高并发访问下,一个热点数据从缓存中不存在,每次请求都要直接查询数据库,从而导致数据库压力过大,并且系统性能下降的现象。

缓存击穿的原因通常有以下几种:

  • 缓存中不存在所需的热点数据:当系统中某个热点数据需要被频繁访问时,如果这个热点数据最开始没有被缓存,那么就会导致系统每次请求都需要直接查询数据库,造成数据库负担。
  • 缓存的热点数据过期:当一个热点数据过期并需要重新缓存时,如果此时有大量请求,那么就会导致所有请求都要直接查询数据库。

在遇到缓存击穿问题时,我们可以在查询数据库之前,先判断一下缓存中是否已有数据,如果没有数据则使用Redis的单线程特性,先查询数据库然后将数据写入缓存中。

org.springframework.bootspring-boot-starter-data-redis

先从缓存中查询数据,如果缓存中无数据则进行锁操作

@Autowired
private RedisTemplate redisTemplate;@GetMapping("/user/{id}")
public User getUserById(@PathVariable Long id){// 先从缓存中获取值String userKey = "user_"+id.toString();User user = (User) redisTemplate.opsForValue().get(userKey);if(user == null){// 查询数据库之前加锁String lockKey = "lock_user_"+id.toString();String lockValue = UUID.randomUUID().toString();try{Boolean lockResult = redisTemplate.opsForValue().setIfAbsent(lockKey, lockValue, 60, TimeUnit.SECONDS);if(lockResult != null && lockResult){// 查询数据库user = userRepository.findById(id).orElse(null);if(user != null){// 将查询到的数据加入缓存redisTemplate.opsForValue().set(userKey, user, 300, TimeUnit.SECONDS);}}}finally{// 释放锁if(lockValue.equals(redisTemplate.opsForValue().get(lockKey))){redisTemplate.delete(lockKey);}}}return user;
}

3.缓存雪崩

缓存中大量数据的失效时间集中在某一个时间段,导致在这个时间段内缓存失效并额外请求数据库查询数据的请求大量增加,从而对数据库造成极大的压力和负荷

  • 缓存服务器宕机:当缓存服务器宕机或重启时,大量的访问请求将直接命中数据库,并在同一时间段内导致大量的数据库查询请求,从而将数据库压力大幅提高。
  • 缓存数据同时失效:在某个特定时间点,缓存中大量数据的失效时间集中在一起,这些数据会在同一时间段失效,并且这些数据被高频访问,将导致大量的访问请求去查询数据库。
  • 缓存中数据过期时间设计不合理:当缓存中的数据有效时间过短,且数据集中在同一时期失效时,就容易导致大量的请求直接查询数据库,加剧数据库压力。
  • 波动式的访问过程:当数据的访问存在波动式特征时,例如输出某些活动物品或促销商品时,将会带来高频的查询请求访问,导致缓存大量失效并产生缓存雪崩。
  1. 一种是将缓存过期时间分散开,即为不同的数据设置不同的过期时间;
  2. 一种是使用Redis的多级缓存架构

添加相关依赖

org.springframework.bootspring-boot-starter-data-redis

net.sf.ehcacheehcache2.10.6

application.properties中配置Ehcache缓存

spring.cache.type=ehcache

创建一个CacheConfig类,用于配置Ehcache

@Configuration
@EnableCaching
public class CacheConfig {@Beanpublic EhCacheCacheManager ehCacheCacheManager(CacheManager cm){return new EhCacheCacheManager(cm);}@Beanpublic CacheManager ehCacheManager(){EhCacheManagerFactoryBean cmfb = new EhCacheManagerFactoryBean();cmfb.setConfigLocation(new ClassPathResource("ehcache.xml"));cmfb.setShared(true);return cmfb.getObject();}
}

在ehcache.xml中添加缓存配置



先从Ehcache缓存中获取,如果缓存中无数据则再从Redis缓存中获取数据

@Autowired
private RedisTemplate redisTemplate;@Autowired
private CacheManager ehCacheManager;@GetMapping("/user/{id}")
@Cacheable(value = "userCache", key = "#id")
public User getUserById(@PathVariable Long id){// 先从Ehcache缓存中获取String userKey = "user_"+id.toString();User user = (User) ehCacheManager.getCache("userCache").get(userKey).get();if(user == null){// 再从Redis缓存中获取user = (User) redisTemplate.opsForValue().get(userKey);if(user != null){ehCacheManager.getCache("userCache").put(userKey, user);}}return user;
}

相关内容

热门资讯

疯抢43轮!溢价33.9%!刚... 就在刚刚,江宁两幅低密地块成功出让!其中,三山G50地块热度非常高,该地块是三山板块近10年来首次推...
57岁湖北前首富骤然身亡,家居... wumiancaijing.com / 几个月前才喊着“既没有退休打算,更没想过逃避责任”的汪林朋...
布米普特拉北京投资基金管理有限... 当地时间七月二十九日,美国液化天然气(LNG)出口领域的新锐巨头Venture Global宣布成功...
国泰A500ETF半年蒸发百亿... 2025年中,公募总规模突破34万亿元,同比增加3万亿,其中指数基金的规模净增了2.17万亿元,是主...
原创 美... 前言 近日,美国财政部长贝森特在瑞典斯德哥尔摩的美中贸易谈判中公开透露。 美国国会正在审议一项针对...
年入250亿,波司登让中产又爱... 文 | 本原财经 中产的心头好,“羽绒服之王”波司登交卷了:2025财年,营收首次达到259.02...
创业板指跌逾2%!下跌个股近4... 【大河财立方消息】7月30日午后,指数走弱,创业板指下挫跌逾2.00%,沪指跌0.16%,深成指跌1...
纺织行业上市公司董秘PK:太平... 来源:新浪证券 数据显示,截止7月29日,A股市场共有5817家上市公司。董秘作为连接投资者与上市公...
文远知行Robotaxi获沙特... 本报讯 (记者袁传玺)7月28日,广州文远知行科技有限公司(以下简称“文远知行”,Nasdaq:WR...
新三板创新层公司九州量子大宗交... 每经讯,2025年7月30日,新三板创新层公司九州量子(837638,收盘价:1.64元)发生一笔大...
美国将对欧盟葡萄酒征收15%关... 经过紧张的谈判和不断升级的猜测,美国最终确认对所有进入美国市场的欧盟商品(包括葡萄酒、烈酒和利口酒)...
阿维塔陈卓:拟明年下半年推出与... 凤凰网科技讯 7月30日,阿维塔科技总裁陈卓在中国长安汽车集团有限公司正式成立后首场媒体沟通会上透露...
累计融资5亿! 广州跑出超级独... 对话 | 铅笔道CEO 王方 撰稿 | 铅笔道编辑 赵松格 外卖赛道有美团,生物医药界也有“美团”?...
美国股市今夏是否会过热?专家建... 财联社7月30日讯(编辑 黄君芝)对金融市场来说,今年夏天是一个复杂的时期。尽管市场一直在猜测未来会...
2025“数通链谷”杯上海市区... 7月30日,2025“数通链谷”杯上海市区块链产业应用职工劳动和技能竞赛在上海市静安区市北高新园区数...
李想回忆被赶出公司:站在对方的... 近日,李想在与鲁豫的对话中,回忆人生低谷:2008年,自己曾被创业伙伴联合逼宫,汽车之家面临严重的财...
中证香港300银行指数报118... 金融界7月30日消息,A股三大指数收盘涨跌不一,中证香港300银行指数 (H300银行,H30324...
凯德北京投资基金管理有限公司:... 7月15日,A股上演惊险过山车行情。沪指早盘冲高至3278点后快速回落,最终微涨0.17%报3241...
美财政部长称中国如继续购买俄罗... 7月30日,外交部发言人郭嘉昆主持例行记者会。有记者问:美国财政部长表示,如果中国继续购买俄罗斯石油...