双重判定锁来解决缓存击穿问题

缓存击穿场景

缓存击穿是指:当热点数据过期或者不存在时,此时正好有大量请求访问热点数据,这些请求就会直接访问数据库,造成数据库的压力骤增,严重时造成宕机。

解决缓存击穿有几种方案:

  1. 缓存预热
    对热点数据进行预加载,当活动开始时,提前将热点数据从数据库加载到缓存中,可以避免海量请求第一次访问热点数据时需要从数据库读取的流程。
  2. 热点数据永不过期
    对于已知的热点数据,设置过期时间为-1,这样就不会有缓存击穿的情况出现了。
  3. 分布式锁
    使用分布式锁,使得只有一个请求可以访问数据库,避免大量请求并发访问数据库。
    但是这种方法存在一定弊端:
    获取分布式锁的请求,都会执行查询数据库并且更新缓存,理论上只有第一次加载数据库请求的记录是有效的。所以这里采用双重判定锁的形势来解决。

    即在获取到分布式锁后,再次进行判断缓存是否存在,如果存在则直接返回。如果不存在,才执行访问数据库并更新缓存的操作

缓存击穿使用 lock 或者 tryLock 同样有所讲究,需要视场景而定:

lock:

获取锁,成功就执行操作,失败就阻塞等待。

trylock:

获取锁,成功返回true,失败则直接返回false,不用阻塞等待。

Lock源码

csharp 复制代码
public interface Lock {
    /**
     * Acquires the lock.
     */
    void lock();
 
    /**
     * Acquires the lock unless the current thread is
     * {@linkplain Thread#interrupt interrupted}.
     */
    void lockInterruptibly() throws InterruptedException;
 
    /**
     * Acquires the lock if it is free within the given waiting time and the
     * current thread has not been {@linkplain Thread#interrupt interrupted}.
     */
    boolean tryLock();
 
    /**
     * Acquires the lock if it is free within the given waiting time and the
     * current thread has not been {@linkplain Thread#interrupt interrupted}.
     */
    boolean tryLock(long time, TimeUnit unit) throws InterruptedException;
 
    /**
     * Releases the lock.
     */
    void unlock();
 
    /**
     * Returns a new {@link Condition} instance that is bound to this
     * {@code Lock} instance.
     */
    Condition newCondition();
}

发生极端情况时,lock和trylock如何选择?

如果希望确保线程能够按照特定的顺序访问共享资源,并且不介意可能的等待时间,那么lock方法是一个不错的选择,但是,如果希望避免线程长时间等待,并且能够处理未能立即获得锁的情况,那么tryLock方法可能更适合。

那么trylcok具体对应于什么样的场景呢?

如果是双十一抢优惠券或者真的是海量访问,在用户占便宜的情况下,推荐使用trylock,因为用户会因为失败而自己刷新页面,海量访问也推荐使用trylock,因为如果使用lock对于用户的体验非常不好。

相关推荐
披着羊皮不是狼6 小时前
(7)为 RAG 系统接入 Redis Stack 实现向量持久化
数据库·redis·缓存
難釋懷7 小时前
数据同步策略
缓存
程序员潘子9 小时前
【保姆级教程】B 站缓存 m4s 文件转 MP4,无损合成一行命令搞定
缓存·ffmpeg·ffmpeg\
Micro麦可乐9 小时前
Redis只会用来做缓存?解锁Redis非缓存的九个应用场景,90%程序员不知道的隐藏技能
数据库·redis·缓存·消息队列·分布式锁·延迟队列·布隆过滤器
键盘鼓手苏苏9 小时前
Flutter 三方库 persistent_cache_simple 的鸿蒙化适配指南 - 实现具备磁盘溢出淘汰与极简 API 的本地持久化缓存、支持端侧资源异步落地与状态秒开实战
flutter·缓存·harmonyos
21号 19 小时前
10.Redis 缓存
数据库·redis·缓存
从零开始的-CodeNinja之路9 小时前
【Redis】Redis 缓存应用、淘汰机制—(四)
java·redis·缓存
星辰徐哥9 小时前
CDN工作原理:节点缓存、智能调度,减少跨网传输延迟
服务器·缓存·php
星辰徐哥9 小时前
ARP缓存表:作用、查看方法与刷新技巧
开发语言·缓存·php
Jul1en_11 小时前
【Redis】String 类型命令、编码方式与应用场景
数据库·redis·缓存