redis分布式锁到redisson的转变

首先导入redis依赖:

<dependency>
     <groupId>org.springframework.boot</groupId>
     <artifactId>spring-boot-starter-data-redis</artifactId>
</dependency>

然后配置redis:

spring:
    redis:
        host: 192.168.1.210
        port: 6379

使用redis实现分布式主体代码:

@Autowired
StringRedisTemplate redisTemplate;

public String getDistributeResult() {
    //1.占分布式锁
    String uuid = UUID.randomUUID().toString();
    Boolean lock = redisTemplate.opsForValue().setIfAbsent("lock", uuid, 300, TimeUnit.SECONDS);//保证锁的赋值和过期时间设置的原子性,同时过期时间避免出现死锁问题
    if(lock){
        System.out.println("获取分布式锁成功。。。");
        //加锁成功...执行业务
        String result = "";
        try{
            result = getResult();
        }finally{
            String script = "if redis.call('get', KEYS[1]) == ARGV[1] then return redis.call('del', KEYS[1]) else return 0 end";
            //删除锁
            Long lock1 = redisTemplate.execute(new DefaultRedisScript<Long>(script, Long.class), Arrays.asList("lock"), uuid);
        }

        return result;
    }else{
        try {
        	//休眠100ms重试
            Thread.sleep(200);
        } catch (InterruptedException e) {

        }
        return getDistributeResult();//自旋方式
    }
}

public String getResult(){
	return "ok";
}
lua脚本解锁(获取值并进行对比)也是为了避免客户端在过期时间后尝试释放锁,删除稍后获得锁的另一个客户端创建的,参考:SET | Redis

开发中经常会使用redisson代替上面繁琐的(容易遗漏的)分布式锁实现,大量节省了开发时间,让程序员把更多的精力放在业务开发上。

想要使用redisson首先要引入依赖(redis的也需要前面已经引入过了)

<dependency>
    <groupId>org.redisson</groupId>
    <artifactId>redisson</artifactId>
    <version>3.12.0</version>
</dependency>

创建redisson自定义配置类:

import org.redisson.Redisson;
import org.redisson.api.RedissonClient;
import org.redisson.config.Config;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;

import java.io.IOException;

@Configuration
public class MyRedissonConfig {

    @Bean
    public RedissonClient redisson() throws IOException {
        //1.创建配置
        Config config = new Config();
        config.useSingleServer().setAddress("redis://192.168.1.210:6379");
        RedissonClient redissonClient = Redisson.create(config);
        return redissonClient;
    }
}

然后就可以使用redisson了,如下:

@Autowired
RedissonClient redisson;

public String getResultWithRedissonLock() {
    RLock lock = redisson.getLock("redisson-lock");
    lock.lock();

    String result = "";
    try{
        result = getResult();
    }finally{
        lock.unlock();
    }

    return result;
}

lock()加锁,unlock()释放锁,就是这么简单。

redisson的锁默认过期时间30s(LockWatchdogTimeout看门狗的默认时间),且支持自动续期(即业务执行未完成可以延长锁的过期时间),只要占锁成功,就会启动一个定时任务【重新给锁设置过期时间,新的过期时间就是看门狗的默认时间】,每隔10s(internalLockLeaseTime【看门狗时间】 / 3, 10s)都会自动再次续期,续成30s

如果我们设置了过期时间,那么就不会存在自动续期等操作,

lock.lock(10, TimeUnit.SECONDS);但是一定要保证自动解锁的时间要大于业务执行的时间。

同时redisson还支持多种锁机制,包含RReadWriteLock,RSemaphore,RCountDownLatch等

可以根据自己的业务场景使用不同的锁。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值