服务器之家:专注于服务器技术及软件下载分享
分类导航

Mysql|Sql Server|Oracle|Redis|MongoDB|PostgreSQL|Sqlite|DB2|mariadb|Access|数据库技术|

服务器之家 - 数据库 - Redis - 总结一下Redis的缓存雪崩、缓存击穿、缓存穿透

总结一下Redis的缓存雪崩、缓存击穿、缓存穿透

2023-05-07 05:02未知服务器之家 Redis

缓存击穿 缓存击穿指的是在高并发情况下,一个缓存的key在缓存中不存在,导致每次请求都要访问数据库,从而导致数据库压力过大,甚至崩溃。这种情况通常发生在一些热点数据上,比如用户登录信息等。 原因 缓存击穿的原因

缓存击穿

缓存击穿指的是在高并发情况下,一个缓存的key在缓存中不存在,导致每次请求都要访问数据库,从而导致数据库压力过大,甚至崩溃。这种情况通常发生在一些热点数据上,比如用户登录信息等。

原因

缓存击穿的原因是因为在某些热点数据的key失效或者被删除时,大量的并发请求同时访问这个key,导致缓存中不存在这个key的数据,从而每个请求都需要去访问数据库获取数据,造成数据库压力过大。

解决方案

1.设置热点数据永不过期

在缓存中设置热点数据永不过期可以有效地避免缓存击穿问题。但是这种方式会导致缓存中存在很多过期但是占用内存的数据,因此需要在设置缓存数据时进行权衡。

String key = "hot_data";
String value = redis.get(key);
if (value == null) {
value = db.get(key);
if (value != null) {
redis.set(key, value);
redis.persist(key); //设置key永不过期
}
}

2.设置热点数据短期过期

为了避免缓存中过多占用内存的数据,可以将热点数据设置一个相对较短的过期时间,比如1分钟,这样可以避免过期数据占用过多内存。当热点数据过期后,可以在后台异步更新缓存数据。

String key = "hot_data";
String value = redis.get(key);
if (value == null) {
//添加分布式锁,避免缓存穿透
if(redis.setNx("lock_"+key,"value")){
value = db.get(key);
if (value != null) {
redis.set(key, value);
redis.expire(key,60); //设置key过期时间为1分钟
}
redis.del("lock_"+key);
}else {
Thread.sleep(50);
return queryDataFromCache(key);
}
}

缓存穿透

缓存穿透指的是当大量的并发请求同时查询一个不存在的key时,由于缓存中没有对应的数据,所以每个请求都会去访问数据库,导致数据库压力过大。

原因

缓存穿透的原因是由于黑客攻击或者恶意请求,可能会对某些不存在的数据进行大量的请求,从而导致缓存穿透问题。

解决方案

1.对查询结果为空的key设置空值

当缓存查询的结果为空时,可以将结果设置为空值写入缓存,这样下次查询相同的key时,可以直接从缓存中获取结果,避免了查询数据库的开销。

String key = "not_exist_data";
String value = redis.get(key);
if (value == null) {
//添加分布式锁,避免缓存穿透
if(redis.setNx("lock_"+key,"value")){
value = db.get(key);
if (value != null) {
redis.set(key, value);
}else {
redis.set(key, ""); //设置空值
redis.expire(key, 60); //设置过期时间为1分钟
}
redis.del("lock_"+key);
}else {
Thread.sleep(50);
return queryDataFromCache(key);
}
}

2.BloomFilter过滤非法请求

使用BloomFilter可以对请求参数进行过滤,将非法请求拦截在系统外部,从而避免了对系统的压力。

BloomFilter filter = new BloomFilter(10000, 0.001); //设置布隆过滤器
String key = "not_exist_data";
if(filter.mightContain(key)){
return null;
}
String value = redis.get(key);
if (value == null) {
//添加分布式锁,避免缓存穿透
if(redis.setNx("lock_"+key,"value")){
value = db.get(key);
if (value != null) {
redis.set(key, value);
}else {
filter.put(key); //将非法key加入过滤器
}
redis.del("lock_"+key);
}else {
Thread.sleep(50);
return queryDataFromCache(key);
}
}

缓存雪崩

缓存雪崩指的是在缓存中存在大量的key过期时间相同或者失效的情况下,当这些key同时失效时,大量的并发请求都会涌入数据库,导致数据库压力过大,甚至崩溃。

原因

缓存雪崩的原因是因为在缓存中存在大量的key同时过期,导致大量的并发请求同时涌入数据库。

解决方案

1.缓存数据随机过期时间 为了避免缓存中大量key同时过期,可以设置每个缓存数据的过期时间不同,比如可以在原有过期时间的基础上添加一个随机时间,这样可以避免大量key同时过期的情况。

String key = "hot_data";
String value = redis.get(key);
if (value == null) {
//添加分布式锁,避免缓存穿透
if(redis.set

2.缓存数据预加载 为了避免在缓存中大量的key失效,可以在缓存数据过期之前,提前将缓存数据刷新到缓存中,保证数据的可用性。

String key = "hot_data";
String value = redis.get(key);
if (value == null) {
//添加分布式锁,避免缓存穿透
if(redis.setNx("lock_"+key,"value")){
value = db.get(key);
if (value != null) {
redis.set(key, value);
redis.expire(key, 1800); //设置过期时间为30分钟
}
redis.del("lock_"+key);
}else {
Thread.sleep(50);
return queryDataFromCache(key);
}
}else {
//判断缓存是否需要刷新
if(redis.ttl(key) < 300){
new Thread(() -> {
String newValue = db.get(key);
if (newValue != null) {
redis.set(key, newValue);
redis.expire(key, 1800); //设置过期时间为30分钟
}
}).start();
}
}

3.限流降级 当缓存雪崩问题出现时,可以通过限流降级的方式来减少对数据库的请求,从而保证系统的可用性。可以通过配置Hystrix等限流降级框架来实现。

String key = "hot_data";
String value = redis.get(key);
if (value == null) {
//使用Hystrix进行限流降级
value = HystrixCommand.execute(() -> {
String data = db.get(key);
redis.set(key, data);
redis.expire(key, 1800); //设置过期时间为30分钟
return data;
}, () -> {
return "系统繁忙,请稍后重试!";
});
}

总结

Redis的使用,可以有效地提高系统的性能和可用性。但是在使用过程中,需要注意缓存击穿、缓存穿透和缓存雪崩等问题,采用适当的解决方案来避免这些问题的发生,从而保证系统的稳定性和可靠性。


延伸 · 阅读

精彩推荐
  • RedisRedis全量复制与部分复制示例详解

    Redis全量复制与部分复制示例详解

    这篇文章主要给大家介绍了关于Redis全量复制与部分复制的相关资料,文中通过示例代码介绍的非常详细,对大家学习或者使用Redis爬虫具有一定的参考学习...

    豆子先生5052019-11-27
  • Redis详解Redis复制原理

    详解Redis复制原理

    与大多数db一样,Redis也提供了复制机制,以满足故障恢复和负载均衡等需求。复制也是Redis高可用的基础,哨兵和集群都是建立在复制基础上实现高可用的...

    李留广10222021-08-09
  • Redisredis 交集、并集、差集的具体使用

    redis 交集、并集、差集的具体使用

    这篇文章主要介绍了redis 交集、并集、差集的具体使用,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友...

    xiaojin21cen10152021-07-27
  • RedisRedis如何实现数据库读写分离详解

    Redis如何实现数据库读写分离详解

    Redis的主从架构,能帮助我们实现读多,写少的情况,下面这篇文章主要给大家介绍了关于Redis如何实现数据库读写分离的相关资料,文中通过示例代码介绍...

    罗兵漂流记6092019-11-11
  • Redisredis实现排行榜功能

    redis实现排行榜功能

    排行榜在很多地方都能使用到,redis的zset可以很方便地用来实现排行榜功能,本文就来简单的介绍一下如何使用,具有一定的参考价值,感兴趣的小伙伴们...

    乘月归5022021-08-05
  • Redisredis中如何使用lua脚本让你的灵活性提高5个逼格详解

    redis中如何使用lua脚本让你的灵活性提高5个逼格详解

    这篇文章主要给大家介绍了关于redis中如何使用lua脚本让你的灵活性提高5个逼格的相关资料,文中通过示例代码介绍的非常详细,对大家的学习或者工作具...

    一线码农5812019-11-18
  • RedisRedis 事务知识点相关总结

    Redis 事务知识点相关总结

    这篇文章主要介绍了Redis 事务相关总结,帮助大家更好的理解和学习使用Redis,感兴趣的朋友可以了解下...

    AsiaYe8232021-07-28
  • RedisRedis的配置、启动、操作和关闭方法

    Redis的配置、启动、操作和关闭方法

    今天小编就为大家分享一篇Redis的配置、启动、操作和关闭方法,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧 ...

    大道化简5312019-11-14