缓存穿透是指数据库和缓存都没有的数据,这样缓存永远不会生效,大量的请求有可能导致数据库宕机。
一般处理缓存穿透有布隆过滤器 和 缓存null值 两种方式。
布隆过滤器是使用一个初始全部是0的位数组,插入元素时先哈希一下,把哈希计算出的多个值的对应位置设为1,然后下次再有同样的元素来时先计算一下,如果对应位置的位都是1那就代表有这个元素。这样再去放行访问reids。但布隆过滤器是存在误判的可能性的,因为它走的是哈希思想,只要哈希思想,就可能存在哈希冲突。
第二种解决缓存穿透的方式就是:缓存空对象的思想比较简单,查询发现缓存和数据库都没有,就给缓存里加一个空值,下次这个请求再来直接返回缓存的空对象就行。但有个问题就是可能会造成短期内数据的不一致,比如缓存空对象的过期时间为10秒,如果在这10秒内底层数据发生了变化,而缓存层的查询仍然会返回缓存的空对象,就会导致短期内数据不一致。
缓存雪崩是指,有很多数据,数据库有,但缓存没有(比如同时失效或者Redis服务宕机),导致这些大量的请求不走redis而是直接去查数据库的情况
多层级缓存,或者也可以给不同的Key的TTL添加随机值
缓存击穿是指,数据库有,但缓存没有的某个热点数据,突然失效,导致的大量这个key请求不走redis而是直接去查数据库的情况。
【之所以会出现这个缓存击穿问题,主要原因是在于我们对key设置了过期时间,假设我们不设置过期时间,其实就不会有缓存击穿的问题,但是不设置过期时间,数据就会一直占用我们内存】
出现这种问题之后,当然是需要把热点数据回写到缓存里,那这就会有并发写的问题。
一般处理方案是1互斥锁,2逻辑过期
只要它们使用同一把锁,就能保障共享资源的正确性和一致性。
互斥锁实现简单,因为仅仅只需要加一把锁,不用其他的操作了,但它只能串行执行,性能肯定受到影响。
逻辑过期指的是,我们把过期时间设置在value中。假设线程1去查询缓存,从value中判断出来当前的数据已经过期了,那它就会去获得互斥锁,然后专门开启一个新线程11去进行重构数据的逻辑,而线程1此时不等了直接返回过期的旧数据,直到新开的线程完成这个逻辑后,才会释放锁。在这个过程当中如果线程2过来访问,由于线程线程11持有着锁,所以线程2无法获得锁,也直接返回旧数据,只有等到新开的线程11把重建数据构建完后,之后其他线程才能返回正确的数据。
逻辑过期的优点在于它更新缓存的操作是异步进行的,其他线程不用等待。缺点在于在构建完缓存之前,返回的都是脏数据。
分布式锁的核心思想在于所有线程都共享同一把锁。