双重检验锁(redis缓存击穿)

1.场景:

用redis做了数据缓存,可能存在缓存击穿,就是同一时间大量请求过来,发现缓存没有数据,大量去查询数据库。

2.解决方案

(判断缓存是否有数据方法外加锁,存在问题后续每次进来都需要判断,影响效率)

2.从数据库查询数据方法外加锁,锁内再次判断缓存是否有数据(双重检验锁)

  //判断缓存中是否有数据
        String skuInfoJson = redisTemplate.opsForValue().get("SkuInfoById" + skuId);
        if(StringUtils.isEmpty(skuInfoJson)){
            //缓存不包含数据,加锁,防止统一时间大量请求进来查询数据库
            synchronized (this){
                //再次判断缓存是否有数据
                String skuInfoJson2 = redisTemplate.opsForValue().get("SkuInfoById" + skuId);
                //没有则查库
                if(StringUtils.isEmpty(skuInfoJson2)){
                    SkuInfoEntity skuInfo = skuInfoService.getById(skuId);
                    redisTemplate.opsForValue().set("SkuInfoById" + skuId,
                            JSONObject.toJSONString(skuInfo), 1, TimeUnit.DAYS);
                    System.out.println("数据库");
                    return R.ok().put("skuInfo",skuInfo);
                }
            }
        }
        //缓存中包含数据,从缓存中取数据
        System.out.println("缓存1");
        SkuInfoEntity skuInfo = JSONObject.parseObject(skuInfoJson, SkuInfoEntity.class);

你可能感兴趣的:(Java开发)