[转载]Redis常见问题 | 缓存穿透、击穿、雪崩、预热、更新、降级及对应的解决方案

Redis是高性能的分布式内存数据库,对于内存数据库经常会出现下面几种情况,也经常会出现在Redis面试题中:缓存穿透、缓存击穿、缓存雪崩、缓存预热、缓存更新、缓存降级。本文分别介绍这些概念以及对应的解决方案。

 

缓存穿透

当查询Redis中没有的数据时,该查询会下沉到数据库层,同时数据库层也没有该数据,当这种情况大量出现或被恶意攻击时,接口的访问全部透过Redis访问数据库,而数据库中也没有这些数据,我们称这种现象为"缓存穿透"。缓存穿透会穿透Redis的保护,提升底层数据库的负载压力,同时这类穿透查询没有数据返回也造成了网络和计算资源的浪费。

解决方案:

 

  • 在接口访问层对用户做校验,如接口传参、登陆状态、n秒内访问接口的次数;

  • 利用布隆过滤器,将数据库层有的数据key存储在位数组中,以判断访问的key在底层数据库中是否存在;

第一种解决方案很好理解,这里介绍一下第二种方案,在前一篇文章中我们介绍了Redis的布隆过滤器,我们知道布隆过滤器可以判断key一定不在集合内以及key极有可能在集合内。

 

基于布隆过滤器,我们可以先将数据库中数据的key存储在布隆过滤器的位数组中,每次客户端查询数据时先访问Redis:

 

  • 如果Redis内不存在该数据,则通过布隆过滤器判断数据是否在底层数据库内;

  • 如果布隆过滤器告诉我们该key在底层库内不存在,则直接返回null给客户端即可,避免了查询底层数据库的动作;

  • 如果布隆过滤器告诉我们该key极有可能在底层数据库内存在,那么将查询下推到底层数据库即可;

 

布隆过滤器有误判率,虽然不能完全避免数据穿透的现象,但已经可以将99.99%的穿透查询给屏蔽在Redis层了,极大的降低了底层数据库的压力,减少了资源浪费。

 

缓存击穿

缓存击穿和缓存穿透从名词上可能很难区分开来,它们的区别是:穿透表示底层数据库没有数据且缓存内也没有数据,击穿表示底层数据库有数据而缓存内没有数据。当热点数据key从缓存内失效时,大量访问同时请求这个数据,就会将查询下沉到数据库层,此时数据库层的负载压力会骤增,我们称这种现象为"缓存击穿"。

解决方案:

 

  • 延长热点key的过期时间或者设置永不过期,如排行榜,首页等一定会有高并发的接口;

  • 利用互斥锁保证同一时刻只有一个客户端可以查询底层数据库的这个数据,一旦查到数据就缓存至Redis内,避免其他大量请求同时穿过Redis访问底层数据库;

 

在使用互斥锁的时候需要避免出现死锁或者锁过期的情况:

 

  • 使用前面文章介绍过的lua脚本或事务将获取锁和设置过期时间作为一个原子性操作(如:set kk vv nx px 30000),以避免出现某个客户端获取锁之后宕机导致的锁不被释放造成死锁现象;

  • 另起一个线程监控获取锁的线程的查询状态,快到锁过期时间时还没查询结束则延长锁的过期时间,避免多次查询多次锁过期造成计算资源的浪费;

缓存雪崩

缓存雪崩是缓存击穿的"大面积"版,缓存击穿是数据库缓存到Redis内的热点数据失效导致大量并发查询穿过redis直接击打到底层数据库,而缓存雪崩是指Redis中大量的key几乎同时过期,然后大量并发查询穿过redis击打到底层数据库上,此时数据库层的负载压力会骤增,我们称这种现象为"缓存雪崩"。事实上缓存雪崩相比于缓存击穿更容易发生,对于大多数公司来讲,同时超大并发量访问同一个过时key的场景的确太少见了,而大量key同时过期,大量用户访问这些key的几率相比缓存击穿来说明显更大。

解决方案:

 

  • 在可接受的时间范围内随机设置key的过期时间,分散key的过期时间,以防止大量的key在同一时刻过期;

  • 对于一定要在固定时间让key失效的场景(例如每日12点准时更新所有最新排名),可以在固定的失效时间时在接口服务端设置随机延时,将请求的时间打散,让一部分查询先将数据缓存起来;

  • 延长热点key的过期时间或者设置永不过期,这一点和缓存击穿中的方案一样;

缓存预热

缓存预热如字面意思,当系统上线时,缓存内还没有数据,如果直接提供给用户使用,每个请求都会穿过缓存去访问底层数据库,如果并发大的话,很有可能在上线当天就会宕机,因此我们需要在上线前先将数据库内的热点数据缓存至Redis内再提供出去使用,这种操作就成为"缓存预热"。

 

缓存预热的实现方式有很多,比较通用的方式是写个批任务,在启动项目时或定时去触发将底层数据库内的热点数据加载到缓存内。

 

缓存更新

缓存服务(Redis)和数据服务(底层数据库)是相互独立且异构的系统,在更新缓存或更新数据的时候无法做到原子性的同时更新两边的数据,因此在并发读写或第二步操作异常时会遇到各种数据不一致的问题。如何解决并发场景下更新操作的双写一致是缓存系统的一个重要知识点。

 

第二步操作异常:缓存和数据的操作顺序中,第二个动作报错。如数据库被更新, 此时失效缓存的时候出错,缓存内数据仍是旧版本;

 

缓存更新的设计模式有四种:

 

Cache aside:查询:先查缓存,缓存没有就查数据库,然后加载至缓存内;更新:先更新数据库,然后让缓存失效;或者先失效缓存然后更新数据库; 

 

Read through:在查询操作中更新缓存,即当缓存失效时,Cache Aside 模式是由调用方负责把数据加载入缓存,而 Read Through 则用缓存服务自己来加载;

 

Write through:在更新数据时发生。当有数据更新的时候,如果没有命中缓存,直接更新数据库,然后返回。如果命中了缓存,则更新缓存,然后由缓存自己更新数据库; 

 

Write behind caching:俗称write back,在更新数据的时候,只更新缓存,不更新数据库,缓存会异步地定时批量更新数据库;

 

 

 

Cache aside:

 

  • 为了避免在并发场景下,多个请求同时更新同一个缓存导致脏数据,因此不能直接更新缓存而是另缓存失效。

  • 先更新数据库后失效缓存:并发场景下,推荐使用延迟失效(写请求完成后给缓存设置1s过期时间),在读请求缓存数据时若redis内已有该数据(其他写请求还未结束)则不更新。当redis内没有该数据的时候(其他写请求已令该缓存失效),读请求才会更新redis内的数据。这里的读请求缓存数据可以加上失效时间,以防第二步操作异常导致的不一致情况。

  • 先失效缓存后更新数据库:并发场景下,推荐使用延迟失效(写请求开始前给缓存设置1s过期时间),在写请求失效缓存时设置一个1s延迟时间,然后再去更新数据库的数据,此时其他读请求仍然可以读到缓存内的数据,当数据库端更新完成后,缓存内的数据已失效,之后的读请求会将数据库端最新的数据加载至缓存内保证缓存和数据库端数据一致性;在这种方案下,第二步操作异常不会引起数据不一致,例如设置了缓存1s后失效,然后在更新数据库时报错,即使缓存失效,之后的读请求仍然会把更新前的数据重新加载到缓存内。

推荐使用先失效缓存,后更新数据库,配合延迟失效来更新缓存的模式;

 

四种缓存更新模式的优缺点:

 

  • Cache Aside:实现起来较简单,但需要维护两个数据存储,一个是缓存(Cache),一个是数据库(Repository);

  • Read/Write Through:只需要维护一个数据存储(缓存),但是实现起来要复杂一些;

  • Write Behind Caching:与Read/Write Through 类似,区别是Write Behind Caching的数据持久化操作是异步的,但是Read/Write Through 更新模式的数据持久化操作是同步的。优点是直接操作内存速度快,多次操作可以合并持久化到数据库。缺点是数据可能会丢失,例如系统断电等。

缓存本身就是通过牺牲强一致性来提高性能,因此使用缓存提升性能,就会有数据更新的延迟性。这就需要我们在评估需求和设计阶段根据实际场景去做权衡了。

 

缓存降级

缓存降级是指当访问量剧增、服务出现问题(如响应时间慢或不响应)或非核心服务影响到核心流程的性能时,即使是有损部分其他服务,仍然需要保证主服务可用。可以将其他次要服务的数据进行缓存降级,从而提升主服务的稳定性。

 

降级的目的是保证核心服务可用,即使是有损的。如去年双十一的时候淘宝购物车无法修改地址只能使用默认地址,这个服务就是被降级了,这里阿里保证了订单可以正常提交和付款,但修改地址的服务可以在服务器压力降低,并发量相对减少的时候再恢复。

 

降级可以根据实时的监控数据进行自动降级也可以配置开关人工降级。是否需要降级,哪些服务需要降级,在什么情况下再降级,取决于大家对于系统功能的取舍。

 

原文链接:https://mp.weixin.qq.com/s/ohg98-PeRo_3AjZ_lp78rw

<think>嗯,用户让我介绍一下基于Redis的高频访问信息缓存,以及如何解决缓存穿透雪崩击穿的问题,并且还要提出相关的面试问题。首先,我需要确保自己对这些概念有清晰的理解,然后才能组织出合适的面试问题。 首先,缓存穿透。我记得这是指查询一个不在的数据,导致每次请求都直接打到数据库上。解决方法可能有布隆过滤器或者缓存空对象。但得确认这些方法的优缺点,比如布隆过滤器的误判率和空间占用,缓存空对象的过期时间设置。 然后是缓存雪崩,这应该是大量缓存同时过期或者Redis宕机,导致数据库压力激增。解决方法可能包括随机过期时间、集群高可用,或者永不过期加后台更新。需要区分雪崩击穿的区别,雪崩是大面积失效,击穿是单个热点数据失效。 缓存击穿,指的是热点数据在过期瞬间,大量请求直接访问数据库。解决方法可能有互斥锁或者逻辑过期。互斥锁可能会影响性能,而逻辑过期需要后台更新,但数据可能不一致。 接下来,用户需要的是面试问题,所以需要从基础到进阶,覆盖原理、解决方案、实际应用场景。可能需要设计问题来考察候选人对这些概念的理解深度,以及解决实际问题的能力。 比如,基础问题可以问缓存穿透雪崩击穿的区别,解决方案。进阶问题可能涉及布隆过滤器的实现原理,互斥锁的具体实现方式,比如Redis的setnx命令。还有实际场景题,让候选人设计一个防止雪崩的方案,或者如何处理热点数据突然过期的情况。 还要注意Redis数据结构,比如如何选择合适的数据结构来缓存,以及持久化策略对缓存的影响。可能还要涉及Redis集群的高可用方案,比如主从复制、哨兵、Cluster模式等。 需要确保问题覆盖全面,既有理论又有实践,同时考察候选人的问题解决能力和经验。比如,是否知道预热缓存降级策略、熔断机制等。 另外,可能还要考虑Redis的内淘汰策略,比如LRU、LFU,不同策略在缓存场景中的应用。以及如何监控缓存的有效性,如何评估缓存命中率等运维相关的问题。 最后,要提醒候选人在回答时结合项目经验,举例说明他们是如何在实际项目中应用这些方案的,以展示实际操作能力。</think>以下是为"Redis缓存高频访问与三⼤缓存问题"主题整理的20个典型⾯试题及考察要点分层说明: --- ### ⼀、基础原理层(5题) 1. **缓存穿透/雪崩/击穿三者的核⼼区别是什么?** - 穿透:**恶意访问不数据** - 雪崩:**⼤⾯积缓存同时失效** - 击穿:**单个热点缓存失效后并发冲击** 2. **描述布隆过滤器防穿透的⼯作原理** - 要点:**位数组+哈希函数、在误判率、空间效率优势** 3. **缓存空对象⽅案如何处理数据更新问题?** - 需答出:**设置较短的过期时间、异步更新策略** 4. **缓存雪崩的随机过期时间⽅案如何实现?** - 公式示例:$$ T_{real} = T_{base} + random(0, T_{variance}) $$ 5. **互斥锁解决击穿问题时,为什么要⽤SETNX⽽不是普通的加锁?** - 关键点:**Redis原⼦性操作、防⽌分布式锁冲突** --- ### ⼆、进阶设计层(8题) 6. **如何设计⼀个带熔断机制的缓存系统?** - 需包含:**请求量阈值监控、降级策略、恢复机制** 7. **逻辑过期⽅案中,如何实现后台异步更新?** - 典型流程:**获取过期标识→触发更新任务→返回旧数据** 8. **布隆过滤器如何应对数据库扩容?** - 难点说明:**需要重建过滤器、采⽤可扩展布隆过滤器** 9. **热点Key发现⽅案有哪些?** - 参考答案:**客户端统计、代理层监控、Redis监控命令** 10. **如何通过双缓存策略防击穿?** - 结构示例: ```text 主缓存:正常过期时间 备份缓存:更⻓过期时间 ``` 11. **Redis持久化策略对缓存恢复的影响?** - 需⽐较:**RDB快照恢复速度 vs AOF数据完整性** 12. **缓存预热的具体实施⽅案?** - 要点:**启动时加载热点数据、历史访问数据分析** 13. **如何通过限流保护数据库?** - 技术栈举例:**Redis-Cell模块、令牌桶算法实现** --- ### 三、综合实战层(7题) 14. **设计⼀个电商秒杀系统的缓存架构** - 考察点:**热点库缓存、本地缓存+Redis多级缓存、库扣减⽅式** 15. **当缓存数据库⼀致性发⽣冲突时如何取舍?** - 需分析:**CAP理论中的CP/AP选择、业务场景特性** 16. **如何优化⼤Value缓存导致的性能问题?** - 解决思路:**数据压缩、分片储、使⽤更合适的数据结构** 17. **处理缓存数据的常⽤⼿段?** - 组合⽅案:**设置合理的过期时间+版本号机制+延迟双删** 18. **如何实现缓存资源的动态扩容?** - 集群⽅案:**Redis Cluster的槽位迁移原理** 19. **设计⼀个实时监控缓存健康度的系统** - 关键指标:**命中率、响应时间、内使⽤率、淘汰率** 20. **当发现缓存命中率持续下降时,你的排查思路?** - 排查路径:**热点Key分析→过期策略检查→业务查询模式变化→内碎⽚情况** --- ### 四、深度扩展题(附加) - **Redis6.0新特性对缓存优化的影响?** - 重点:**多线程IO、客户端缓存(Client-side caching)** - **如何应⽤机器学习预测缓存失效?** - 创新点:**基于历史访问模式的智能过期时间调整** --- 以上问题设计覆盖**原理理解→⽅案设计→系统架构→故障排查**的知识链条,建议候选⼈在准备时: 1. 结合具体项⽬经验说明 2. 对每个解决⽅案要能说明优缺点 3. 重点掌握布隆过滤器、分布式锁、熔断降级等核⼼技术点的实现细节
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值