用redis实现分布式锁

Posted zzbjava

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了用redis实现分布式锁相关的知识,希望对你有一定的参考价值。

  用一个redis 实现分布式锁。

1.为什么要用reids 实现分布式锁(比较于syschronized 的优势) : 既然是分布式锁于syschronized 的优势 就在于分布式可用。java 提供的syschronized,lock(syschronized,lock内存语义不同.以后会写)等但是只适用于一个服务器。

 

www.redis.cn 中文redis网站。

 

本次实现redis分布式锁的几个  redis 命令。

SETNX :key设置值为value,如果key不存在,这种情况下等同SET命令。 当key存在时,什么也不做。SETNX是”SET if Not eXists”的简写

GETSET:自动将key对应到value并且返回原来key对应的value。如果key存在但是对应的value不是字符串,就返回错误。

 

代码:

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;

@Component
@Slf4j
public class RedisLock {

@Autowired
private StringRedisTemplate redisTemplate;

/**
* @param key 订单id
* @param value 当前时间+超时时间
* @return
*/
public synchronized boolean lock(String key,String value){
//对应的是reids 中的SETNX命令 设置成功为true
if(redisTemplate.opsForValue().setIfAbsent(key, value)){
return true;
}
      
      //下面的意义是解决 当我们加锁后由于代码或者网络的原因中间抛出异常未能正常解锁
String currentValue = redisTemplate.opsForValue().get(key);

if(!StringUtils.isEmpty(currentValue) && Long.parseLong(currentValue)<System.currentTimeMillis()){

//获取上一个锁的时间
String oldValue = redisTemplate.opsForValue().getAndSet(key, value);
if(oldValue != null && oldValue.equals(currentValue)){
return true;
}
}

return false;
}

/**
* 解锁
* @param key
* @param value
*/
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 );
}


}
}

 

我用的是springBoot 2.0.5版本的

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

<[email protected]依赖--> 
<dependency>
<groupId>org.projectlombok</groupId>
<artifactId>lombok</artifactId>
</dependency>

 

 

 

GETSET





































































以上是关于用redis实现分布式锁的主要内容,如果未能解决你的问题,请参考以下文章

用redis实现分布式锁

用Redis实现分布式锁 与 实现任务队列

用Redis实现分布式锁 与 实现任务队列

编程实践分布式锁的实现代码

基于 Redis 分布式锁实现“秒杀”(含代码)

java 使用 Redis 实现分布式锁