ToB企服应用市场:ToB评测及商务社交产业平台

标题: Redis的缓存穿透、缓存击穿和缓存雪崩 [打印本页]

作者: 八卦阵    时间: 2024-7-28 21:31
标题: Redis的缓存穿透、缓存击穿和缓存雪崩
目次
一、解释阐明
二、缓存穿透
 1. 什么是缓存穿透?
 2. 常见的两种解决方案
 (1)缓存空对象
 (2)布隆过滤
3. 编码解决商品查询的缓存穿透题目
三、缓存击穿
 1.  什么是缓存击穿?
 2、缓存击穿解决方案(2种)
(1)互斥锁
(2)逻辑逾期
 3.  互斥锁与逻辑逾期的对比分析​编辑
 四、利用互斥锁解决缓存击穿题目
(1)首先,我们声明一下获取锁、释放锁的方法,tryLock()、unLock()
(2)互斥锁解决缓存击穿 queryWithMutex() 
五、利用逻辑逾期解决缓存击穿题目
(1)添加逻辑逾期时间的字段
(2)逻辑逾期解决缓存击穿题目 queryWithLogicalExpire()
 六、缓存雪崩
1.  什么是缓存雪崩?
2.  缓存雪崩解决方案(4种)
七、封装 Redis 工具类


一、解释阐明

   Redis缓存穿透、缓存击穿和缓存雪崩都是缓存机制中的一些题目,具体解释如下:
  1. (1)缓存穿透(Cache Penetration):指查询一个不存在的数据,由于缓存中没有数据,
  2.              所以这个查询请求会直接穿过缓存层,到达数据库层,造成了数据库的压力。
  3.              攻击者可以通过构造恶意请求,使得缓存层无法命中任何数据,
  4.              从而导致请求直接访问数据库,从而引起数据库压力过大。
  5. (2)缓存击穿(Cache Breakdown):指缓存中某个热点数据失效,此时有大量并发请求同时访问
  6.              这个失效的数据,导致这些请求直接访问数据库,造成数据库压力过大,
  7.              甚至导致数据库崩溃。通常是由于缓存中某个热点数据过期失效,
  8.              同时有大量并发请求访问该数据。
  9. (3)缓存雪崩(Cache Avalanche):指缓存中大量的数据失效,导致大量请求直接访问数据库,
  10.              造成数据库压力过大。通常是由于缓存中大量的数据在同一时间失效,
  11.              导致大量请求直接访问数据库。
复制代码
  针对上述题目,可以采取以下步伐:
  1. (1)缓存穿透:可以在查询缓存之前,先对请求的参数进行合法性检查,如过滤非法字符、
  2.               判断参数范围等;或者使用BloomFilter等数据结构,对查询参数进行过滤,
  3.               只有在BloomFilter中判断有可能存在的情况下才会去查询数据库。
  4. (2)缓存击穿:可以使用锁机制或者分布式锁机制,避免大量并发请求同时访问失效的热点数据。
  5.               或者不设置TTL,设置逻辑上过期标识,需要过期的时候直接删除标识
  6.             
  7. (3)缓存雪崩:可以采用多级缓存架构,减少缓存层的压力;
  8.               或者设置热点数据的过期时间为随机时间,避免在同一时间大量数据同时失效。
  9.               另外可以在缓存层和数据库层之间添加限流、熔断等措施,
  10.               避免因突发流量导致系统崩溃。
复制代码
二、缓存穿透

1. 什么是缓存穿透?

    缓存穿透是指客户端哀求的数据在缓存中和数据库中都不存在,如许缓存永久不会生效,这些哀求都会打到数据库
2. 常见的两种解决方案

(1)缓存空对象

  1. 简单的来说,就是请求之后,发现数据不存在,就将null值打入Redis中。
  2. 优点:实现简单,维护方便
  3. 缺点:额外的内存消耗
  4.       可能造成短期的不一致
复制代码
  1. 思路分析:
  2.         当我们客户端访问不存在的数据时,先请求 redis,但是此时 redis 中没有数据,
  3. 此时会访问到数据库,但是数据库中也没有数据,这个数据穿透了缓存,直击数据库,
  4. 我们都知道数据库能够承载的并发不如 redis 这么高,如果大量的请求同时过来访问这种不存在的数据,
  5. 这些请求就都会访问到数据库,简单的解决方案就是哪怕这个数据在数据库中也不存在,
  6. 我们也把这个数据存入到 redis 中去,这样,下次用户过来访问这个不存在的数据,
  7. 那么在 redis 中也能找到这个数据就不会进入到数据库了。
复制代码


 (2)布隆过滤

  1.      在客户端与Redis之间加了一个布隆过滤器,对请求进行过滤。
  2.      布隆过滤器的大致的原理:布隆过滤器中存放二进制位。
  3.                数据库的数据通过hash算法计算其hash值并存放到布隆过滤器中,
  4.                之后判断数据是否存在的时候,就是判断该hash值是0还是1。
  5.                但是这是一种概率上的统计,当其判断不存在的时候就一定是不存在;
  6.                 当其判断存在的时候就不一定存在。所以有一定的穿透风险
复制代码
  1. 优点:内存占用较少,没有多余 key
  2. 缺点:实现复杂 存在误判可能
复制代码


综上所述

               我们可以两种方案一起用,如许子最为保险。据统计使用布隆过滤器一般可以避免90%的无效哀求。
 3. 编码解决商品查询的缓存穿透题目

  1. 核心思路如下:
  2.             在原来的逻辑中,我们如果发现这个数据在 mysql 中不存在,直接就返回 404 了,
  3. 这样是会存在缓存穿透问题的
  4. 现在的逻辑中:
  5.             如果这个数据不存在,我们不会返回 404 ,还是会把这个数据写入到 Redis 中,
  6. 并且将 value 设置为空,当再次发起查询时,我们如果发现命中之后,判断这个 value 是否是 null,
  7. 如果是 null,则是之前写入的数据,证明是缓存穿透数据,如果不是,则直接返回数据。
复制代码

我们在这里只要做两件事:
  1. 当查询数据在数据库中不存在时,将空值写入 redis
  2.  判断缓存是否命中后,再加一个判断是否为空值
复制代码
  1. @Override
  2. public Result queryById(Long id) {
  3.     // 从redis查询商铺缓存
  4.     String key = CACHE_SHOP_KEY + id;
  5.     String shopJson = stringRedisTemplate.opsForValue().get(key);
  6.     // 判断是否存在
  7.     if (StrUtil.isNotBlank(shopJson)) {
  8.         // 存在,直接返回
  9.         Shop shop = JSONUtil.toBean(shopJson, Shop.class);
  10.         return Result.ok(shop);
  11.     }
  12.     // 1.判断空值
  13.     if (shopJson.isBlank(shopJson)) {
  14.         // 返回一个错误信息
  15.         return Result.fail("店铺不存在!");
  16.     }
  17.     // 不存在,根据id查询数据库
  18.     Shop shop = getById(id);
  19.     // 不存在,返回错误
  20.     if (shop == null) {
  21.         
  22.         // 2.防止穿透问题,将空值写入redis!!!
  23.         stringRedisTemplate.opsForValue().set(key,"",CACHE_NULL_TTL,TimeUnit.MINUTES);
  24.         return Result.fail("店铺不存在!");
  25.     }
  26.     // 存在,写入Redis
  27.     // 把shop转换成为JSON形式写入Redis
  28.     // 同时添加超时时间
  29.     stringRedisTemplate.opsForValue().set(key, JSONUtil.toJsonStr(shop), CACHE_SHOP_TTL, TimeUnit.MINUTES);
  30.     return Result.ok(shop);
  31. }
复制代码
总结:
缓存穿透产生的缘故原由是什么?
  1. 用户请求的数据在缓存中和数据库中都不存在,不断发起这样的请求,给数据库带来巨大压力
复制代码
缓存穿透的解决方案有哪些?
  1. (1)缓存 null 值
  2. (2)布隆过滤
  3. (3)增强 id 的复杂度,避免被猜测 id 规律
  4. (4)做好数据的基础格式校验
  5. (5)加强用户权限校验
  6. (6)做好热点参数的限流
复制代码
三、缓存击穿

 1.  什么是缓存击穿?

缓存击穿是部分key逾期导致的严重结果
  1. 为什么大量key过期会产生问题而少量的key也会有问题?
  2.     缓存击穿问题也叫热点Key问题,就是⼀个被高并发访问并且缓存重建业务较复杂的key突然失效了,
  3. 无数的请求访问会在瞬间给数据库带来巨大的冲击。
复制代码
具体情况如下图所示:

 上述:假设此时该热点key的TTL时间到(失效了),则查询缓存未掷中,会继续查询数据库,并进行缓存重修工作但是由于查询SQL逻辑比较复杂、重修缓存的时间较久,而且该key又是热点key,短时间内有大量的线程对其进行访问,所以哀求会直接 “打到” 数据库中,数据库就有可能崩掉
2、缓存击穿解决方案(2种)

(1)互斥锁

  1. 简单的来说:
  2.           并不是所有的线程都有 “ 资格 ” 去访问数据库,只有持有锁的线程才可以对其进行操作。
  3. 不过该操作有一个很明显的问题,就是会出现相互等待的情况。
复制代码


  (2)逻辑逾期

  1. 不设置TTL
  2.          之前所说导致缓存击穿的原因就是该key的TTL到期了,所以我们在这就不设置TTL了,
  3. 而是使用一个字段,例如:expire表示过期时间(逻辑上的)。当我们想让它 “ 过期 ” 的时候,
  4. 我们可以直接手动将其删除(热点key,即只是在一段时间内,其被访问的频次很高)。
  5. 这种方案巧妙在于,异步的构建缓存,缺点在于在构建完缓存之前,返回的都是脏数据。
复制代码


 3.  互斥锁与逻辑逾期的对比分析


 四、利用互斥锁解决缓存击穿题目

  1.  核心思路:
  2.          相较于原来从缓存中查询不到数据后直接查询数据库而言,现在的方案是 进行查询之后,
  3. 如果从缓存没有查询到数据,则进行互斥锁的获取,获取互斥锁后,判断是否获得到了锁,如果没有得到,
  4. 则休眠,过一会再进行尝试,直到获取到锁为止,才能进行查询
  5.          如果获取到了锁的线程,再去进行查询,查询后将数据写入 redis,再释放锁,返回数据,
  6. 利用互斥锁就能保证只有一个线程去执行操作数据库的逻辑,防止缓存击穿。
复制代码


代码实现

(1)首先,我们声明一下获取锁、释放锁的方法,tryLock()、unLock()

  1. /**
  2.   * 获取锁
  3.   * @param key
  4.   * @return
  5. */
  6. private boolean tryLock(String key) {
  7.     // setnx 就是 setIfAbsent 如果存在
  8.     Boolean flag = stringRedisTemplate.opsForValue().setIfAbsent(key, "1", 10, TimeUnit.MINUTES);
  9.     // 装箱是将值类型装换成引用类型的过程;拆箱就是将引用类型转换成值类型的过程
  10.     // 不要直接返回flag,可能为null
  11.     return BooleanUtil.isTrue(flag);
  12. }
  13. /**
  14. * 释放锁
  15. * @param key
  16. */
  17. private void unLock(String key) {
  18.     stringRedisTemplate.delete(key);
  19. }
复制代码
留意:这里的锁不是真正的线程锁,而是redis里面的一个特殊的key。
(2)互斥锁解决缓存击穿 queryWithMutex() 

  1. /**
  2. * 互斥锁解决缓存击穿 queryWithMutex()
  3. * @param id
  4. * @return
  5. */
  6. public Shop queryWithMutex(Long id) {
  7.     // 1.从redis查询商铺缓存
  8.     String key = CACHE_SHOP_KEY + id;
  9.     String shopJson = stringRedisTemplate.opsForValue().get(key);
  10.     // 2.判断是否存在
  11.     if (StrUtil.isNotBlank(shopJson)) {
  12.         return JSONUtil.toBean(shopJson, Shop.class);
  13.     }
  14.     // 判断空值
  15.     if (shopJson != null) {
  16.         // 返回一个错误信息
  17.         return null;
  18.     }
  19.     String lockKey = "lock:shop:" + id;
  20.     Shop shop = null;
  21.     try {
  22.         // 4.实现缓存重建
  23.         // 4.1获取互斥锁
  24.         boolean isLock = tryLock(lockKey);
  25.         // 4.2判断是否成功
  26.         if (!isLock) {
  27.             // 4.3失败,则休眠并重试
  28.             Thread.sleep(50);
  29.             // 递归
  30.             return queryWithMutex(id);
  31.         }
  32.         // 4.4成功,根据id查询数据库
  33.         shop = getById(id);
  34.         // 模拟延迟
  35.         Thread.sleep(200);
  36.         // 5.不存在,返回错误
  37.         if (shop == null) {
  38.             stringRedisTemplate.opsForValue().set(key,"",CACHE_NULL_TTL,TimeUnit.MINUTES);
  39.             return null;
  40.         }
  41.         // 6.存在,写入redis
  42.         stringRedisTemplate.opsForValue().set(key,JSONUtil.toJsonStr(shop),CACHE_SHOP_TTL,TimeUnit.MINUTES);
  43.     } catch (InterruptedException ex) {
  44.         throw new RuntimeException(ex);
  45.     } finally {
  46.         // 7.释放锁
  47.         unLock(lockKey);
  48.     }
  49.     // 8.返回
  50.     return shop;
  51. }
复制代码
五、利用逻辑逾期解决缓存击穿题目

需求:修改根据id查询商铺的业务,基于逻辑逾期方式来解决缓存击穿题目


留意:这里的key是否逾期,不是由redis控制的,而是由我们本身去手动编写逻辑去控制的。 
代码实现

(1)添加逻辑逾期时间的字段

之前的Shop中是没有逻辑逾期的字段,要如何让它带有这个属性,又不修改之前的代码呢?
新建一个RedisData对象,里面的data指的是Shop对象,而expireTime是逻辑逾期时间。
即:我们可以使用 JSONUtil.toBean 将Shop对象通过序列化、反序列化到RedisData类的data属性中。
  1. @Data
  2. public class RedisData {
  3.     // LocalDateTime : 同时含有年月日时分秒的日期对象
  4.     // 并且LocalDateTime是线程安全的!
  5.     private LocalDateTime expireTime;
  6.     private Object data;
  7. }
复制代码
(2)逻辑逾期解决缓存击穿题目 queryWithLogicalExpire()

缓存重修
  1. /**
  2. * 重建缓存,先缓存预热一下,否则queryWithLogicalExpire() 的expire为null
  3. * @param id
  4. * @param expireSeconds
  5. */
  6. public void saveShopRedis(Long id, Long expireSeconds) {
  7.     // 1.查询店铺数据
  8.     Shop shop = getById(id);
  9.     // 2.封装逻辑过期时间
  10.     RedisData redisData = new RedisData();
  11.     redisData.setData(shop);
  12.     redisData.setExpireTime(LocalDateTime.now().plusSeconds(expireSeconds));  // 过期时间
  13.     // 3.写入redis
  14.     stringRedisTemplate.opsForValue().set(CACHE_SHOP_KEY + id, JSONUtil.toJsonStr(redisData));
  15. }
复制代码
先使用测试方法运行一下saveShopRedis(),否则redis里面没有expireTime !


  1. private static final ExecutorService CACHE_REBUILD_EXECUTOR = Executors.newFixedThreadPool(10);
  2. /**
  3. * 逻辑过期解决缓存击穿问题 queryWithLogicalExpire()
  4. * 测试前要先缓存预热一下!不然 data 与 expireTime 的缓存值是null!
  5. * @param id
  6. * @return
  7. */
  8. public Shop queryWithLogicalExpire(Long id) {
  9.     // 1.从redis查询商铺缓存
  10.     String key = CACHE_SHOP_KEY + id;
  11.     String shopJson = stringRedisTemplate.opsForValue().get(key);
  12.     // 2.判断是否存在
  13.     if (StrUtil.isBlank(shopJson)) {
  14.         return null;
  15.     }
  16.     // 4.命中,需要将json反序列化为对象
  17.     // redisData没有数据
  18.     RedisData redisData = JSONUtil.toBean(shopJson, RedisData.class);
  19.     Shop shop = JSONUtil.toBean((JSONObject) redisData.getData(), Shop.class);
  20.     LocalDateTime expireTime = redisData.getExpireTime();
  21.     // 5.判断是否过期
  22.     if (expireTime.isAfter(LocalDateTime.now())) {
  23.         // 5.1未过期,直接返回店铺信息
  24.         return shop;
  25.     }
  26.     // 5.2已过期,需要缓存重建
  27.     // 6.缓存重建
  28.     // 6.1.获取互斥锁
  29.     String lockKey = LOCK_SHOP_KEY + id;
  30.     boolean islock = tryLock(lockKey);
  31.     // 6.2.判断是否获取互斥锁成功
  32.     if (islock) {
  33.         // 6.3.成功,开启独立线程,实现缓存重建
  34.         CACHE_REBUILD_EXECUTOR.submit( () -> {
  35.             try {
  36.                 // 重建缓存,过期时间为20L
  37.                 saveShopRedis(id,20L);
  38.             } catch (Exception ex) {
  39.                 throw new RuntimeException(ex);
  40.             } finally {
  41.                 unLock(lockKey);
  42.             }
  43.         });
  44.     }
  45.     // 6.4.返回过期店铺信息
  46.     return shop;
  47. }
复制代码
可以看到在测试的时间,name的值为:“100XXXX”
修改一下数据库,将值改为:“900XXXX”,看看并发情况下缓存重修能否正确

 通过Jmeter做压力测试

 再查看Redis中的数据,可以看到name的值已经被修改了,而且上面的jmeter的每一个http都是正常的!


 六、缓存雪崩

1.  什么是缓存雪崩?

        缓存雪崩是指在同一时段大量的缓存key同时失效或者Redis服务宕机,导致大量哀求到达数据库,带来巨大压力。
  1. 缓存击穿是部分key过期导致的严重后果,而缓存雪崩则是因为大量的key同时过期所导致的问题
复制代码
情况大抵如下图所示:



2.  缓存雪崩解决方案(4种)

  1. (1)给不同的Key的TTL添加随机值(推荐)
  2.         操作简单,当我们在做缓存预热的时候,就有可能在同一时间批量插入大量的数据,
  3. 那么如果它们的TTL都一样的话就可能出现大量key同时过期的情况!!!
  4. 所以我们需要在设置过期时间TTL的时候,定义一个范围,追加该范围内的一个随机数。
  5. (2)利用Redis集群提高服务的可用性
  6.         使用集群提高可靠性
  7. (3)给缓存业务添加降级限流策略
  8.         微服务的知识
  9. (4)给业务添加多级缓存  
  10.         请求到达浏览器,nginx可以做缓存,未命中找Redis,再未命中找JVM,最后到数据库......
复制代码

七、封装 Redis 工具类

基于 StringRedisTemplate 封装一个缓存工具类,满足下列需求:
  1. 方法 1:将任意 Java 对象序列化为 json 并存储在 string 类型的 key 中,并且可以设置 TTL 过期时间
  2. 方法 2:将任意 Java 对象序列化为 json 并存储在 string 类型的 key 中,并且可以设置逻辑过期时间
复制代码
存击穿题目
  1. 方法 3:根据指定的 key 查询缓存,并反序列化为指定类型,利用缓存空值的方式解决缓存穿透问题
  2. 方法 4:根据指定的 key 查询缓存,并反序列化为指定类型,利用逻辑过期解决缓存击穿问题
复制代码
将逻辑进行封装
  1. @Slf4j
  2. @Component
  3. public class CacheClient {
  4.     private final StringRedisTemplate stringRedisTemplate;
  5.     private static final ExecutorService CACHE_REBUILD_EXECUTOR = Executors.newFixedThreadPool(10);
  6.     public CacheClient(StringRedisTemplate stringRedisTemplate) {
  7.         this.stringRedisTemplate = stringRedisTemplate;
  8.     }
  9.     public void set(String key, Object value, Long time, TimeUnit unit) {
  10.         stringRedisTemplate.opsForValue().set(key, JSONUtil.toJsonStr(value), time, unit);
  11.     }
  12.     public void setWithLogicalExpire(String key, Object value, Long time, TimeUnit unit) {
  13.         // 设置逻辑过期
  14.         RedisData redisData = new RedisData();
  15.         redisData.setData(value);
  16.         redisData.setExpireTime(LocalDateTime.now().plusSeconds(unit.toSeconds(time)));
  17.         // 写入Redis
  18.         stringRedisTemplate.opsForValue().set(key, JSONUtil.toJsonStr(redisData));
  19.     }
  20.     public <R,ID> R queryWithPassThrough(
  21.             String keyPrefix, ID id, Class<R> type, Function<ID, R> dbFallback, Long time, TimeUnit unit){
  22.         String key = keyPrefix + id;
  23.         // 1.从redis查询商铺缓存
  24.         String json = stringRedisTemplate.opsForValue().get(key);
  25.         // 2.判断是否存在
  26.         if (StrUtil.isNotBlank(json)) {
  27.             // 3.存在,直接返回
  28.             return JSONUtil.toBean(json, type);
  29.         }
  30.         // 判断命中的是否是空值
  31.         if (json != null) {
  32.             // 返回一个错误信息
  33.             return null;
  34.         }
  35.         // 4.不存在,根据id查询数据库
  36.         R r = dbFallback.apply(id);
  37.         // 5.不存在,返回错误
  38.         if (r == null) {
  39.             // 将空值写入redis
  40.             stringRedisTemplate.opsForValue().set(key, "", CACHE_NULL_TTL, TimeUnit.MINUTES);
  41.             // 返回错误信息
  42.             return null;
  43.         }
  44.         // 6.存在,写入redis
  45.         this.set(key, r, time, unit);
  46.         return r;
  47.     }
  48.     public <R, ID> R queryWithLogicalExpire(
  49.             String keyPrefix, ID id, Class<R> type, Function<ID, R> dbFallback, Long time, TimeUnit unit) {
  50.         String key = keyPrefix + id;
  51.         // 1.从redis查询商铺缓存
  52.         String json = stringRedisTemplate.opsForValue().get(key);
  53.         // 2.判断是否存在
  54.         if (StrUtil.isBlank(json)) {
  55.             // 3.存在,直接返回
  56.             return null;
  57.         }
  58.         // 4.命中,需要先把json反序列化为对象
  59.         RedisData redisData = JSONUtil.toBean(json, RedisData.class);
  60.         R r = JSONUtil.toBean((JSONObject) redisData.getData(), type);
  61.         LocalDateTime expireTime = redisData.getExpireTime();
  62.         // 5.判断是否过期
  63.         if(expireTime.isAfter(LocalDateTime.now())) {
  64.             // 5.1.未过期,直接返回店铺信息
  65.             return r;
  66.         }
  67.         // 5.2.已过期,需要缓存重建
  68.         // 6.缓存重建
  69.         // 6.1.获取互斥锁
  70.         String lockKey = LOCK_SHOP_KEY + id;
  71.         boolean isLock = tryLock(lockKey);
  72.         // 6.2.判断是否获取锁成功
  73.         if (isLock){
  74.             // 6.3.成功,开启独立线程,实现缓存重建
  75.             CACHE_REBUILD_EXECUTOR.submit(() -> {
  76.                 try {
  77.                     // 查询数据库
  78.                     R newR = dbFallback.apply(id);
  79.                     // 重建缓存
  80.                     this.setWithLogicalExpire(key, newR, time, unit);
  81.                 } catch (Exception e) {
  82.                     throw new RuntimeException(e);
  83.                 }finally {
  84.                     // 释放锁
  85.                     unlock(lockKey);
  86.                 }
  87.             });
  88.         }
  89.         // 6.4.返回过期的商铺信息
  90.         return r;
  91.     }
  92.     public <R, ID> R queryWithMutex(
  93.             String keyPrefix, ID id, Class<R> type, Function<ID, R> dbFallback, Long time, TimeUnit unit) {
  94.         String key = keyPrefix + id;
  95.         // 1.从redis查询商铺缓存
  96.         String shopJson = stringRedisTemplate.opsForValue().get(key);
  97.         // 2.判断是否存在
  98.         if (StrUtil.isNotBlank(shopJson)) {
  99.             // 3.存在,直接返回
  100.             return JSONUtil.toBean(shopJson, type);
  101.         }
  102.         // 判断命中的是否是空值
  103.         if (shopJson != null) {
  104.             // 返回一个错误信息
  105.             return null;
  106.         }
  107.         // 4.实现缓存重建
  108.         // 4.1.获取互斥锁
  109.         String lockKey = LOCK_SHOP_KEY + id;
  110.         R r = null;
  111.         try {
  112.             boolean isLock = tryLock(lockKey);
  113.             // 4.2.判断是否获取成功
  114.             if (!isLock) {
  115.                 // 4.3.获取锁失败,休眠并重试
  116.                 Thread.sleep(50);
  117.                 return queryWithMutex(keyPrefix, id, type, dbFallback, time, unit);
  118.             }
  119.             // 4.4.获取锁成功,根据id查询数据库
  120.             r = dbFallback.apply(id);
  121.             // 5.不存在,返回错误
  122.             if (r == null) {
  123.                 // 将空值写入redis
  124.                 stringRedisTemplate.opsForValue().set(key, "", CACHE_NULL_TTL, TimeUnit.MINUTES);
  125.                 // 返回错误信息
  126.                 return null;
  127.             }
  128.             // 6.存在,写入redis
  129.             this.set(key, r, time, unit);
  130.         } catch (InterruptedException e) {
  131.             throw new RuntimeException(e);
  132.         }finally {
  133.             // 7.释放锁
  134.             unlock(lockKey);
  135.         }
  136.         // 8.返回
  137.         return r;
  138.     }
  139.     private boolean tryLock(String key) {
  140.         Boolean flag = stringRedisTemplate.opsForValue().setIfAbsent(key, "1", 10, TimeUnit.SECONDS);
  141.         return BooleanUtil.isTrue(flag);
  142.     }
  143.     private void unlock(String key) {
  144.         stringRedisTemplate.delete(key);
  145.     }
  146. }
复制代码


免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!更多信息从访问主页:qidao123.com:ToB企服之家,中国第一个企服评测及商务社交产业平台。




欢迎光临 ToB企服应用市场:ToB评测及商务社交产业平台 (https://dis.qidao123.com/) Powered by Discuz! X3.4