微服务:Redis高并发下分布式锁实现

传统的软件公司,依然坚持用着三层架构,单体的应用程序可以使用synchronized解决高并发下秒杀功能,这样程序在并发下访问效率下降,若用nginx反向代理启用分布式应用,synchronized则实现不了分布式锁,如何解决?利用redis中的setnx可实现高并发下分布式锁。具体要点和细节记录在代码注释中,代码如下:

//简单的秒杀功能:

 private final StringRedisTemplate stringRedisTemplate;

 @Autowired
 public MiaoShaController(StringRedisTemplate stringRedisTemplate) {
     this.stringRedisTemplate = stringRedisTemplate;
 }

 @GetMapping("deduct_stock")
 public String deductStock() {
     ValueOperations<String, String> stringStringValueOperations = stringRedisTemplate.opsForValue();
     String lockKey = "lockKey";
     String uid = UUID.randomUUID().toString();
     try {
     //利用redis中的setnx实现抢购物品数量上锁
         Boolean lockKey1 = stringStringValueOperations.setIfAbsent(lockKey, uid);
         stringRedisTemplate.expire(lockKey, 30, TimeUnit.SECONDS);
         if (lockKey1) {
             int goodsCount = NumberUtils.toInt(stringStringValueOperations.get("goodsCount"));
             if (goodsCount > 0) {
                 int count = goodsCount - 1;
                 stringStringValueOperations.set("goodsCount", count + "");
                 System.out.println("秒杀成功,库存:" + count);
             } else {
                 System.out.println("秒杀失败");
             }
         }
        //无论业务代码中是否抛出异常,最终都要释放锁。
     } finally {
     	//解释下if的判断,利用uid当作value做判断的原因是:高并发下,
     	//如果业务代码执行超过30秒,则此时锁被释放,下一个线程进来获取到锁,最终以下代码释放的就是下一个线程锁,导致业务异常。
         if (StringUtils.equals(uid, stringStringValueOperations.get(lockKey))) {
             stringRedisTemplate.delete(lockKey);
         }
     }
     return "end";
 }

高并发下一般使用redisson实现分布式锁,获取锁时默认设置过期时间是30秒,若过期时间超过10秒则重新刷新过期时间为30秒,直到释放锁为止。

...代码后续补充。

你可能感兴趣的:(微服务总结)