Java基于redis实现分布式锁(SpringBoot)
前⾔
分布式锁,其实原理是就是多台机器,去争抢⼀个资源,谁争抢成功,那么谁就持有了这把锁,然后去执⾏后续的业务逻辑,执⾏完毕后,把锁释放掉。
可以通过多种途径实现分布式锁,例如利⽤数据库(mysql等),插⼊⼀条记录(唯⼀索引),谁插⼊成功,谁就持有锁;还可通过zookeeper来实现分布式锁,谁创建节点成功,谁就持有锁。本⽂介绍通过redis来实现分布式锁。
本⽂使⽤springboot提供的RedisTemplate来操作redis,可以参考我之前的⽂章,这⾥对使⽤RedisTemplate来操作redis做了介绍。当然也可以直接使⽤jedis来操作redis,⼤家可以参考下jedis的⽂档,使⽤上都是⼤同⼩异的。
实现分布式锁的步骤
第⼀步:通过redis的setnx⽅式(不存在则设置),往redis上设置⼀个带有过期时间的key,如果设置成功,则获得了分布式锁。这⾥设置过期时间,是防⽌在释放锁的时候出现异常导致锁释放不掉。
第⼆步:执⾏完业务操作之后,删除该锁。
实现
新建⼀个DistributedLock.class,注⼊StringRedisTemplate。
@Component
public class DistributedLock {
@Autowired
private StringRedisTemplate redisTemplate;
}
获得锁
long spring是什么意思
/**
* 获得锁
*/
public boolean getLock(String lockId, long millisecond) {
Boolean success = redisTemplate.opsForValue().setIfAbsent(lockId, "lock",
millisecond, TimeUnit.MILLISECONDS);
return success != null && success;
}
setIfAbsent⽅法,就是当键不存在的时候,设置,并且该⽅法可以设置键的过期时间。该⽅法对应到redis的原⽣命令就是:
SET lockId content PX millisecond NX
⾄于设置多少的过期时间合适,这个是没有定论的,需要根据真是的业务场景来衡量。
释放锁
当处理完业务逻辑后,需要⼿动的把锁释放掉。
public void releaseLock(String lockId) {
redisTemplate.delete(lockId);
}
释放锁的操作⽐较简单,直接删除之前设置的键即可。其实,基于redis实现分布式锁的⽅式,在释放锁的时候,是存在释放失败的风险的(⽐如⽹路抖动什么的),这也是为什么在设置锁的时候需要设置过期时间的原因,可以防⽌在出现异常的时候,锁会⾃动的消失掉。同时,我们也可以增加⼏次失败之后的重试机制。
测试
新建⼀个BusinessTask.java,代码如下:
@Component
public class BusinessTask {
private final static String LOCK_ID = "happyjava";
@Autowired
DistributedLock distributedLock;
@Scheduled(cron = "0/10 * * * * ? ")
public void doSomething() {
boolean lock = Lock(LOCK_ID, 10 * 1000);
if (lock) {
System.out.println("执⾏任务");
} else {
System.out.println("没有抢到锁");
}
}
}
这⾥使⽤了springboot的Scheduled注解来实现定时任务,该cron表达式的意思是每10秒钟,执⾏⼀次任务,然后我们启动两次该项⽬,观察⼀段时间执⾏结果:
第⼀个springboot任务:
第⼆个springboot任务:
两个任务在交替的执⾏任务,证明了同⼀时刻只有⼀个应⽤持有了锁。
总结
本⽂主要介绍了如何使⽤Java代码(springboot的restTemplate)实现Redis分布式锁,对于加锁和解锁也分别给出了⽰例代码。其实我们还可以尝试使⽤Redisson实现分布式锁,这是Redis官⽅提供的Java组件,这个后续再介绍吧。

版权声明:本站内容均来自互联网,仅供演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系QQ:729038198,我们将在24小时内删除。