记录一次synchronized 针对代码块加对象锁案例

一,场景 redis双检锁

并发访问,获取redis缓存时,先查询缓存,无则查询数据库。

存在一个问题,当并发量比较大时,都查询缓存无数据,都会怼到数据库中,造成数据库压力。

二,代码如下

分析一下代码块中部分逻辑

1.再次查询缓存

2.查询数据库,写入缓存

执行流程

假如1000个请求过来,调用1000次CacheTemplateService.fetchCache,都没有查询到缓存数据,将要执行代码块逻辑,因为加了关键字synchronized,1000个请求代码顺序执行,第一个请求进入,再次查询缓存还是没有查询到数据,查库,写缓存。如果再过来1000个请求,第一次查询缓存,就可以查到数据,不会执行代码块逻辑(主要目的)。

这时原来先进入到1000个请求中的后续999个在排队的请求,会依次执行代码块,先查缓存,再查库,试想一下,如果代码块中没有再次查询缓存的逻辑,还是有大量请求怼到数据库,这就是为啥加双检锁。

@Slf4j
@Component
public class CacheTemplateService {

    @Autowired
    private RedisManager redisManager;

    public <T> T fetchCache(String key, Integer expire, TypeReference<T> clazz, CacheLoadable<T> cacheLoadable) {
        String json = (String) redisManager.get(key);
        if (Objects.nonNull(json) && StringUtils.isNotBlank(json) && !json.equalsIgnoreCase("null")
                && !"[]".equals(json) && !"{}".equals(json)) {
            return JSON.parseObject(json, clazz);
        }
        synchronized (this) {
            json = (String) redisManager.get(key);
            if (Objects.nonNull(json) && StringUtils.isNotBlank(json) && !json.equalsIgnoreCase("null")
                    && !"[]".equals(json) && !"{}".equals(json)) {
                return JSON.parseObject(json, clazz);
            }
            // 核心业务
            T result = cacheLoadable.load();
            redisManager.put(key, expire, JSON.toJSONString(result));
            return result;
        }
    }

    public void invalidate(String key) {
        redisManager.remove(key);
    }

}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值