目录
缓存雪崩
一、什么是缓存雪崩?
这是当许多或者大部分的缓存在同一时间失效,导致大量的请求直接访问数据库。这种情况下的流量激增可能会给数据库带来极大的压力,甚至使其崩溃。
二、如何避免缓存雪崩问题?
- 将缓存失效时间随机打散: 我们可以在原有的失效时间基础上增加一个随机值(比如 1 到 10 分钟)这样每个缓存的过期时间都不重复了,也就降低了缓存集体失效的概率。
- 设置缓存不过期: 我们可以通过后台服务来更新缓存数据,从而避免因为缓存失效造成的缓存雪崩,也可以在一定程度上避免缓存并发问题。
缓存击穿
一、什么是缓存击穿?
这是与缓存穿透有些相似但又有所区别的情况。缓存击穿是指一个热点的key失效了,但这个key在数据库中有数据。当大量的并发请求都针对这个失效的key时,它们都无法从缓存中获取数据,因此全部打到了数据库上,导致数据库压力剧增。
二、如何避免击穿问题?
- 互斥锁方案(Redis 中使用 setNX 方法设置一个状态位,表示这是一种锁定状态),保证同一时间只有一个业务线程请求缓存,未能获取互斥锁的请求,要么等待锁释放后重新读取缓存,要么就返回空值或者默认值。
- 不给热点数据设置过期时间,由后台异步更新缓存,或者在热点数据准备要过期前,提前通知后台线程更新缓存以及重新设置过期时间;
缓存穿透
一、什么是缓存穿透?
请求的数据既不在缓存中,也不在数据库中。这意味着每次这样的请求都会直接打到数据库上,造成不必要的压力。缓存穿透可能会对数据库产生巨大的流量,特别是当有恶意用户连续发送大量此类请求时。
二、如何避免穿透问题?
- 非法请求的限制:当有大量恶意请求访问不存在的数据的时候,也会发生缓存穿透,因此在 API 入口处我们要判断求请求参数是否合理,请求参数是否含有非法值、请求字段是否存在,如果判断出是恶意请求就直接返回错误,避免进一步访问缓存和数据库。
- 设置空值或者默认值:当我们线上业务发现缓存穿透的现象时,可以针对查询的数据,在缓存中设置一个空值或者默认值,这样后续请求就可以从缓存中读取到空值或者默认值,返回给应用,而不会继续查询数据库。
- 使用布隆过滤器快速判断数据是否存在,避免通过查询数据库来判断数据是否存在:我们可以在写入数据库数据时,使用布隆过滤器做个标记,然后在用户请求到来时,业务线程确认缓存失效后,可以通过查询布隆过滤器快速判断数据是否存在,如果不存在,就不用通过查询数据库来判断数据是否存在,即使发生了缓存穿透,大量请求只会查询 Redis 和布隆过滤器,而不会查询数据库,保证了数据库能正常运行,Redis 自身也是支持布隆过滤器的。