redis使用分布式锁
在什么样的情况下我们会使用redis的分布式锁
我们举出下面展示一些 内联代码片
。
@Service
public class StockService_lock_syn {
@Autowired
private StockDao stockDao;
public static Object o=new Object();
Lock lock=new ReentrantLock();
public String jianStock(Integer pid){
try {
lock.lock();//加锁
//1. 查询指定的商品库存
Stock stock = stockDao.selectById(pid);
if (stock.getNum() > 0) {
//2.库存减1
stock.setNum(stock.getNum() - 1);
stockDao.updateById(stock);
System.out.println("库存剩余数量:" + stock.getNum());
return "减库存成功";
} else {
System.out.println("库存不足");
return "库存减失败";
}
}
finally {
lock.unlock(); //释放锁
}
}
}
同一个库存数被多个线程卖,线程安全问题
上面的synchronized或Lock锁是否适合集群模式|分布式系统。不适合、因为synchronized都是基于JVM的本地锁。
需要在idea中跑项目的集群
使用用测压软件进行测试,结果出现两台集群出现重卖问题。
我们使用分布式锁来解决
@Autowired
private StringRedisTemplate redisTemplate;
/**
* 2.0redis分布式锁
*/
public String jianStock(Integer pid){
//占锁
ValueOperations<String, String> forValue = redisTemplate.opsForValue();
// Boolean aBoolean = forValue.setIfAbsent("product::" + pid, "", 30, TimeUnit.SECONDS);
//占锁失败
while (!forValue.setIfAbsent("product::" + pid, "", 30, TimeUnit.SECONDS)){
try {
Thread.sleep(20);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
//占锁成功
try {
//1. 查询指定的商品库存
Stock stock = stockDao.selectById(pid);
if (stock.getNum() > 0) {
//2.库存减1
stock.setNum(stock.getNum() - 1);
stockDao.updateById(stock);
System.out.println("库存剩余数量:" + stock.getNum());
return "减库存成功";
} else {
System.out.println("库存不足");
return "库存减失败";
}
}finally {
//释放锁资源
redisTemplate.delete("product::"+pid);
}
}
如果你的业务代码的执行时间超过30s,当前线程删除的是其他线程的锁资源。 --watchDog机制
每个10s检测当前线程是否还持有所资源,如果持有则为当前线程延迟。—可以自己设置watchDog机制,—第三方Redission完美的解决分布式锁。
4.3redisson完美解决redis超时问题
依赖
(2)main函数
@Bean //创建redisson交于spring容器来管理
public RedissonClient redisson() {
Config config = new Config();
config.setLockWatchdogTimeout(300);
config.useSingleServer().setAddress("redis://192.168.107.223:6379");
RedissonClient redisson = Redisson.create(config);
return redisson;
(3)使用
@Autowired
private RedissonClient redisson;
/**
* 3.0
* @param pid
* @return
*/
public String jianStock(Integer pid){
RLock lock = redisson.getLock("product::" + pid);
try {
lock.lock(30,TimeUnit.SECONDS);//加锁: 如果程序执行是出现一次
//1. 查询指定的商品库存
Stock stock = stockDao.selectById(pid);
if (stock.getNum() > 0) {
//2.库存减1
stock.setNum(stock.getNum() - 1);
stockDao.updateById(stock);
System.out.println("库存剩余数量:" + stock.getNum());
return "减库存成功";
} else {
System.out.println("库存不足");
return "库存减失败";
}
}finally {
lock.unlock();
}
}