redis使用lua脚本实现分布式锁

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值