缓存穿透
查询一个不存在的数据,mysql查询不到也不会直接写入缓存,就会导致每次请求都会查找数据库。
解决方案:
1.缓存空数据,若查询结果为空,则把空对象写入缓存
优点:简单
缺点:可能会发送不一致的问题
2.布隆过滤器
优点: 内存占用较少,没有多余key
缺点:实现复杂,存在误判
布隆过滤器:
bitmap(位图):相当于是一个以(bit)位为单位的数组,数组中每个单元只能存储0或1
作用:可以用于检索一个元素是否在一个集合中
存储数据:将一个数据,通过多次hash函数获取hash值,根据hash计算数组对应位置改为1
查询数据:使用相同hash函数获取hash值,判断值是否都为1
误判率:数组越大,误判率越低,同时带来了更多的内存消耗
缓存击穿
给某一个key设置了过期时间,当key过期的时候,恰好这时间点对这个key有大量的并发请求过来,这些并发的请求可能会把DB压垮(热点事件)
解决方案:
1.互斥锁 保证数据的强一致性,性能差 (在获取互斥锁失败时,不让其他线程获取数据)
2.逻辑过期(不设置过期时间) 高可用,性能优,不能保证数据绝对一致 (让其他线程在获取互斥锁失败时,先获取过期数据)
缓存雪崩
在同一时间大量的缓存key同时失效或者Redis服务宕机,导致大量请求到达数据库,带来巨大压力。
解决方案:
1.给不同的Key的TTL设置随机值
2.利用Redis集群提高服务的可用性 哨兵模式、集群模式
3.给缓存业务添加降级限流策略 nginx 或者 spring cloud gateway
4.给业务添加多级缓存 Guava 或 Caffeine
降级可以作为系统的保底策略,适用于穿透、击穿、雪崩