(记录)Java基于redis实现分布式锁(SpringBoot)

前言

分布式锁,其实原理是就是多台机器,去争抢一个资源,谁争抢成功,那么谁就持有了这把锁,然后去执行后续的业务逻辑,执行完毕后,把锁释放掉。

可以通过多种途径实现分布式锁,例如利用数据库(mysql等),插入一条记录(唯一索引),谁插入成功,谁就持有锁;还可通过zookeeper来实现分布式锁,谁创建节点成功,谁就持有锁。本文介绍通过redis来实现分布式锁。

实现分布式锁的步骤

第一步:通过redis的setnx方式(不存在则设置),往redis上设置一个带有过期时间的key,如果设置成功,则获得了分布式锁。这里设置过期时间,是防止在释放锁的时候出现异常导致锁释放不掉。

第二步:执行完业务操作之后,删除该锁。

实现

新建一个DistributedLock.class,注入StringRedisTemplate。

@Component
public class DistributedLock {

    @Autowired
    private StringRedisTemplate redisTemplate;


    /**
     * 获得锁
     */
    public boolean getLock(String lockId, long millisecond) {
        Boolean success = redisTemplate.opsForValue().setIfAbsent(lockId, "lock",
                millisecond, TimeUnit.MILLISECONDS);
        return success != null && success;
    }


    /**
     * 释放锁
     */
    public void releaseLock(String lockId) {
        redisTemplate.delete(lockId);
    }

}

获得锁

setIfAbsent方法,就是当键不存在的时候,设置,并且该方法可以设置键的过期时间。该方法对应到redis的原生命令就是:

SET lockId content PX millisecond NX 

至于设置多少的过期时间合适,这个是没有定论的,需要根据真是的业务场景来衡量。

释放锁

当处理完业务逻辑后,需要手动的把锁释放掉。
释放锁的操作比较简单,直接删除之前设置的键即可。其实,基于redis实现分布式锁的方式,在释放锁的时候,是存在释放失败的风险的(比如网路抖动什么的),这也是为什么在设置锁的时候需要设置过期时间的原因,可以防止在出现异常的时候,锁会自动的消失掉。同时,我们也可以增加几次失败之后的重试机制。

测试

@Component
public class BusinessTask {private final static String LOCK_ID = "happyjava";@Autowired
 DistributedLock distributedLock;@Scheduled(cron = "0/10 * * * * ? ")
 public void doSomething() {
 boolean lock = distributedLock.getLock(LOCK_ID, 10 * 1000);
 if (lock) {
 System.out.println("执行任务");
 distributedLock.releaseLock(LOCK_ID);
 } else {
 System.out.println("没有抢到锁");
 }
 }}

这里使用了springboot的Scheduled注解来实现定时任务,该cron表达式的意思是每10秒钟,执行一次任务,然后我们启动两次该项目,观察一段时间执行结果:

多个任务在交替的执行任务,证明了同一时刻只有一个应用持有了锁。

相应了解:

Redisson,redis lua,KafkaConsumerService

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
下面是使用 Spring Boot 基于 Redis 实现分布式的示例代码: 首先在 pom.xml 中添加 Redis 相关依赖: ```xml <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-data-redis</artifactId> </dependency> ``` 然后创建一个 Redis 分布式的工具类: ```java @Component public class RedisLockUtil { @Autowired private RedisTemplate<String, String> redisTemplate; /** * 尝试获取分布式 * @param key 的名称 * @param value 的值 * @param expireTime 的过期时间,单位为秒 * @return 是否成功获取 */ public boolean tryLock(String key, String value, long expireTime) { Boolean result = redisTemplate.opsForValue().setIfAbsent(key, value, Duration.ofSeconds(expireTime)); return result != null ? result : false; } /** * 释放分布式 * @param key 的名称 * @param value 的值 */ public void unlock(String key, String value) { String currentValue = redisTemplate.opsForValue().get(key); if (StringUtils.isNotBlank(currentValue) && StringUtils.equals(currentValue, value)) { redisTemplate.delete(key); } } } ``` 在上述代码中,我们使用了 RedisTemplate 对 Redis 进行操作,其中 tryLock() 方法尝试获取分布式,如果获取成功则返回 true,否则返回 false。unlock() 方法释放分布式。 最后,在需要使用分布式的地方注入 RedisLockUtil 并使用它进行加和解,示例如下: ```java @Service public class UserService { @Autowired private RedisLockUtil redisLockUtil; public void updateUser(User user) { String lockKey = "updateUser:" + user.getId(); String lockValue = UUID.randomUUID().toString(); try { boolean lockResult = redisLockUtil.tryLock(lockKey, lockValue, 60); if (lockResult) { // 获取分布式成功,执行更新操作 // ... } else { // 获取分布式失败,抛出异常或者重试 // ... } } finally { // 释放分布式 redisLockUtil.unlock(lockKey, lockValue); } } } ``` 在上述代码中,我们调用 RedisLockUtil 的 tryLock() 方法尝试获取分布式,如果获取成功则执行更新操作,否则抛出异常或者重试。最后在 finally 块中调用 RedisLockUtil 的 unlock() 方法释放分布式
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值