<meta name="referrer" content="no-referrer" /> ## 理论知识html
redis分布式锁的实现方案请参考文章 如何优雅地用redis实现分布式锁java
以秒杀活动为例子,在多线程高并发的状况下须要保证秒杀业务的线程安全性,确保秒杀记录与所扣库存数量想匹配。mysql
<small>该段代码能够解决理论知识中的各类问题,包括锁住的时候出现异常,死锁等(经过比较时间戳的方式)</small>redis
import lombok.extern.slf4j.Slf4j; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.data.redis.core.StringRedisTemplate; import org.springframework.stereotype.Component; import org.springframework.util.StringUtils; /** * Redis分布式锁的实现 * @author : wang zns * @date : 2019-05-10 */ @Component @Slf4j public class RedisLock { @Autowired private StringRedisTemplate redisTemplate; /** * 加锁 * @param key seckillId * @param value 当前时间+超时时间 * @return */ public boolean lock(String key, String value) { // 能够设置返回true Boolean isLock = redisTemplate.opsForValue().setIfAbsent(key, value); if (isLock) { return true; } String currentValue = redisTemplate.opsForValue().get(key); // 若是锁已通过期 if (!StringUtils.isEmpty(currentValue) && Long.valueOf(currentValue) < System.currentTimeMillis()) { // 获取上一个锁的时间,并设置新锁的时间 String oldValue = redisTemplate.opsForValue().getAndSet(key, value); if (!StringUtils.isEmpty(oldValue) && oldValue.equals(currentValue)) { log.info("锁过时并返回true"); return true; } } return false; } /** * 解锁 * @param key * @return */ public void unlock(String key, String value) { try { String currentValue = redisTemplate.opsForValue().get(key); if (!StringUtils.isEmpty(currentValue) && currentValue.equals(value)) { redisTemplate.opsForValue().getOperations().delete(key); } } catch (Exception e) { log.error("redis分布式锁,解锁异常, {}",e.getMessage()); } } }
<small>只有当线程拿到锁的时候才可执行try中的业务代码,而且在finally中咱们对锁进行释放</small>spring
long currentTimeMills = System.currentTimeMillis(); String redisLockValue = String.valueOf(currentTimeMills + RedisConstants.LOCK_EXPIRE_TIME); final boolean lock = redisLock.lock(String.valueOf(seckillId), redisLockValue); if (!lock) { throw new RuntimeException("人数过多,请稍后再试"); } try { SeckillExecution execution = secPressureTestService.executeSeckill(seckillId); return Result.success(execution); } catch (Exception e) { log.error("【执行秒杀错误】,message={}", e.getMessage()); return Result.error(e.getMessage()); } finally { redisLock.unlock(String.valueOf(seckillId), redisLockValue); }
ab -n 100 -c 4 127.0.0.1/kill/1000/executionsql
在加上锁以后所扣库存与秒杀记录的数量保持了一致,说明咱们的锁是生效的,你能够不加锁的状况下进行压测,看看结果就能看出差别。apache
若是你在加上了redis分布式锁以后压测结果有问题(所扣库存与秒杀记录不一致),那么有多是锁和mysql事务的提交顺序的问题致使的,解决方法请参考文章java中锁与@Transactional同时使用致使锁失效的问题安全
redis分布式锁在springboot中的使用就介绍到这里,从一个案例入手进行了介绍。虽然咱们还有其余的方式能够达到相同的效果,好比说使用同步锁synchronized关键字,可是通过测试后发现synchronized锁住整个方法的时候性能极差,而且synchronized在分布式系统中的表现不如redis分布式锁。redis分布式锁的强大之处在于分布式,由于线程是否能拿到锁取决于redis的key和value, 这个key和value对于分布式系统是能够共享的,因此redis分布式锁在分布式系统中极为方便和强大。springboot