传统的软件公司,依然坚持用着三层架构,单体的应用程序可以使用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秒,直到释放锁为止。
...代码后续补充。