• 缓存穿透、雪崩与击穿


    1、缓存穿透

    缓存穿透是指数据库和缓存都没有的数据,这样缓存永远不会生效,大量的请求有可能导致数据库宕机。

    强调都没有数据+并发访问

    一般处理缓存穿透有布隆过滤器 和 缓存null值 两种方式。

    布隆过滤器

    布隆过滤器是使用一个初始全部是0的位数组,插入元素时先哈希一下,把哈希计算出的多个值的对应位置设为1,然后下次再有同样的元素来时先计算一下,如果对应位置的位都是1那就代表有这个元素。这样再去放行访问reids。但布隆过滤器是存在误判的可能性的,因为它走的是哈希思想,只要哈希思想,就可能存在哈希冲突。

    缓存NULL值

    第二种解决缓存穿透的方式就是:缓存空对象的思想比较简单,查询发现缓存和数据库都没有,就给缓存里加一个空值,下次这个请求再来直接返回缓存的空对象就行。但有个问题就是可能会造成短期内数据的不一致,比如缓存空对象的过期时间为10秒,如果在这10秒内底层数据发生了变化,而缓存层的查询仍然会返回缓存的空对象,就会导致短期内数据不一致。

    2、缓存雪崩

    缓存雪崩是指,有很多数据,数据库有,但缓存没有(比如同时失效或者Redis服务宕机),导致这些大量的请求不走redis而是直接去查数据库的情况

    强调批量Key过期+并发访问

    多层级缓存,或者也可以给不同的Key的TTL添加随机值

    3、缓存击穿

    缓存击穿是指,数据库有,但缓存没有的某个热点数据,突然失效,导致的大量这个key请求不走redis而是直接去查数据库的情况。

    强调单个Key过期+并发访问

    【之所以会出现这个缓存击穿问题,主要原因是在于我们对key设置了过期时间,假设我们不设置过期时间,其实就不会有缓存击穿的问题,但是不设置过期时间,数据就会一直占用我们内存】
    出现这种问题之后,当然是需要把热点数据回写到缓存里,那这就会有并发写的问题。
    一般处理方案是1互斥锁,2逻辑过期

    互斥锁

    只要它们使用同一把锁,就能保障共享资源的正确性和一致性。
    互斥锁实现简单,因为仅仅只需要加一把锁,不用其他的操作了,但它只能串行执行,性能肯定受到影响。

    逻辑过期

    逻辑过期指的是,我们把过期时间设置在value中。假设线程1去查询缓存,从value中判断出来当前的数据已经过期了,那它就会去获得互斥锁,然后专门开启一个新线程11去进行重构数据的逻辑,而线程1此时不等了直接返回过期的旧数据,直到新开的线程完成这个逻辑后,才会释放锁。在这个过程当中如果线程2过来访问,由于线程线程11持有着锁,所以线程2无法获得锁,也直接返回旧数据,只有等到新开的线程11把重建数据构建完后,之后其他线程才能返回正确的数据。
    逻辑过期的优点在于它更新缓存的操作是异步进行的,其他线程不用等待。缺点在于在构建完缓存之前,返回的都是脏数据。

    分布式并发控制

    分布式锁的核心思想在于所有线程都共享同一把锁。

    • 实现的方式是使用redis中的 SET NX命令获取锁,这个命令可以保证互斥性,只有一个线程能够成功获取锁。
      为了避免死锁情况的发生,在线程拿锁的同时也设置锁的过期时间,这样即使系统发生故障也能保证锁在一定时间后自动释放。
    • 还有就是在分布式条件下,可能会出现一些极端情况,【A线程拿到锁然后阻塞了,时间到了之后锁过期被放掉。然后B线程获取到互斥锁开始执行逻辑,A线程这时候恢复了,又继续执行然后把B的锁释放掉】。那为了解决这种情况拿锁的时候存进自己的线程标识,在释放锁时,先验证一下这把锁是不是自己存的,确定后才删除,这样就不会出现删除别的线程锁的情况。
    • 除此之外,会有一个拿锁-比锁-释放的过程,为了避免出现在这个过程还没走完系统宕机或者其他极端情况影响系统的可靠性,拿锁-比锁-释放的代码封装到一个lua脚本里,这样代码执行的时候就能保证它的原子性。
      基于setnx实现的分布式锁存在下面的问题:
      1、主从一致性问题: 如果Redis提供了主从集群,当我们向集群写数据时,主机需要异步的将数据同步给从机,而万一在同步过去之前,主机宕机了,就会出现数据不一致问题。
      2、可重入问题:同一个线程在持有锁的情况下可以继续执行需要获取同一锁的代码,而不会被阻塞。可重入锁有助于避免死锁和提高代码的灵活性。(lua里拿锁,当锁已经存在时,判断传入的线程标识是否相等,如果相等代表就是可重入锁返回1代表获取到了锁,并且重置过期时间)
  • 相关阅读:
    研发效能|DevOps 是运维还是开发?
    Shopro商城 高级版 Fastadmin和Uniapp进行开发的多平台商城(微信公众号、微信小程序、H5网页、Android-App、IOS-App)
    备战金九银十!2022Java面试必刷461道大厂架构面试真题汇总+面经+简历模板都放这了,注意划重点!!
    open ai chartgpt 安装插件 txyz.ai
    MQ通道接收端绑定步骤
    大气污染数据系统
    5G/4G/北斗遥测终端机全国各省水利平台无缝对接
    C++ 就地构造对象
    【工作中问题解决实践 六】基于反射及类装饰模式的数据对比框架
    java学生综合能力的教学评估系统ssm
  • 原文地址:https://blog.csdn.net/weixin_44104529/article/details/139984635