Redis分布式锁

- redis

代码实现分布式锁

springboot依赖

<!-- StringRedisTemplate需要 -->
<dependency>
  <groupId>org.springframework.boot</groupId>
  <artifactId>spring-boot-starter-data-redis</artifactId>
</dependency>
<!-- @slf4j注解使用 -->
<dependency>
    <groupId>org.projectlombok</groupId>
    <artifactId>lombok</artifactId>
    <version>1.16.6</version>
</dependency>

实现逻辑

方法:返回值true代表获取锁成功,返回值false则相反

方法:参数value=当前时间+超时时间

方法:参数key为redis的键

  1. 判断key是否已存在,不存在就新增,返回true,已存在不改变原有key的value,返回false
  2. 获取key对应的cuurentValue(设value时间+超时时间)
  3. 判断锁是否超过有效期, 是否有效 = 有效期(value) < 系统当前时间
  4. 如果没有过期证明锁还在被使用,直接返回false,如果已经过期,说明锁还在被使用,并且超过了有效期,所以执行第五步
  5. 通过 getAndSet 先返回key的值,并且给key设置新的值
  6. 判断旧的值与过了有效期的值是否相等,如果等于就返回true,否则返回false

flow

代码

package com.xumf.service;

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
     * @param value 当前时间+超时时间
     * @return
     */
    public boolean lock(String key, String value) {
        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 (!StringUtils.isEmpty(oldValue) && 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("解锁异常,{}", e);
        }
    }
}

redisson框架实现



maven依赖


<!-- JDK 1.8+ 依赖 -->
<dependency>
   <groupId>org.redisson</groupId>
   <artifactId>redisson</artifactId>
   <version>3.6.5</version>
</dependency>

<!-- JDK 1.6+ 依赖 -->
<dependency>
   <groupId>org.redisson</groupId>
   <artifactId>redisson</artifactId>
   <version>2.11.5</version>
</dependency><!--