package com.paic.phucp.console.djss;
import com.paic.phucp.common.utils.RedisUtil;
import org.springframework.stereotype.Component;
import redis.clients.jedis.JedisCluster;
import java.util.Collections;
/**
* 第一个为key,我们使用key来当锁名
* 第二个为value,传的是uid,唯一随机数,也可以使用本机mac地址 + uuid
* 第三个为NX,意思是SET IF NOT EXIST,即当key不存在时,我们进行set操作;若key已经存在,则不做任何操作
* 第四个为PX,意思是我们要给这个key加一个过期的设置,具体时间由第五个参数决定 第五个为time,代表key的过期时间,对应第四个参数 PX毫秒,EX秒
* */
@Component
public class LockImpl {
public boolean lock(String lockId, String lockValue, int timeoutInSeconds) {
JedisCluster jedisCluster = RedisUtil.getJedisCluster();
String result = jedisCluster.set(lockId, lockValue, "NX", "EX", timeoutInSeconds);
return "OK".equals(result);
}
public boolean unlock(String lockId, String lockValue) {
JedisCluster jedisCluster = RedisUtil.getJedisCluster();
String script = "if redis.call('get', KEYS[1]) == ARGV[1] then return redis.call('del', KEYS[1]) else return 0 end";
Object result = jedisCluster.eval(script, Collections.singletonList(lockId), Collections.singletonList(lockValue));
return (result instanceof Long) && ((Long) result).longValue() == 1;
}
}
redis分布式锁使用lua脚本是为了保证原子性
redis分布式锁用来做job可能会出现的问题:上一次Job还没执行完,锁已经失效了,下一次JOB又触发了,那么两个JOB并发执行,可能出现业务问题,解决方案是使用Redisson或者Zookeeper分布式锁
缓存必须设置过期时间,这是一种规范。如果不过期,大家都设置,最后内存满了,那么该删除什么数据呢?
Redisson简介
Redisson是分布式协调Redis客服端,帮助用户在分布式环境中轻松实现一些Java的对象,Redisson、Jedis、Lettuce是三个不同的操作Redis的客户端,Redisson侧重分布式开发,其余两个侧重CRUD,底层使用netty框架
Redisson分布式锁原理:看门狗模式,通过异步线程,netty心跳,实现锁自动续期
调用Redisson的API进行加解锁,命令类似redis