Redis实现分布式锁,非lua脚本

Redis实现分布式锁,非lua脚本

初始化Spring StringRedisTemplate

LettuceConnectionFactory factory = new LettuceConnectionFactory();
factory.setHostName("localhost");
factory.setPort(6379);
factory.afterPropertiesSet();
StringRedisTemplate stringRedisTemplate = new StringRedisTemplate();
stringRedisTemplate.setConnectionFactory(factory);
stringRedisTemplate.afterPropertiesSet();

//清空上次测试数据
stringRedisTemplate.delete("lock:uid1");
//锁id
String lockId = UUID.randomUUID().toString();

加锁

stringRedisTemplate.execute(new RedisCallback<Void>() {
    @Override
    public Void doInRedis(RedisConnection connection) throws DataAccessException {
        RedisSerializer<String> keySerializer = (RedisSerializer<String>) stringRedisTemplate.getKeySerializer();
        RedisSerializer<String> valueSerializer = (RedisSerializer<String>) stringRedisTemplate.getValueSerializer();
        byte[] key = keySerializer.serialize("lock:uid1");
        byte[] value = valueSerializer.serialize(lockId);
        Long ttl = connection.ttl(key, TimeUnit.SECONDS);
        if (-1 == ttl) {
            //清理异常超时
            connection.del(key);
        }
        //setNX将key的值设为value,当且仅当key不存在。
        //若给定的key已经存在,则SETNX不做任何动作。
        Boolean ret = connection.setNX(key, value);
        if (ret) {
            //setNX成功,设置锁的超时时间3秒
            connection.expire(key, 3);
        }
        return null;
    }
});

解锁

stringRedisTemplate.execute(new RedisCallback<Void>() {
    @Override
    public Void doInRedis(RedisConnection connection) throws DataAccessException {
        RedisSerializer<String> keySerializer = (RedisSerializer<String>) stringRedisTemplate.getKeySerializer();
        RedisSerializer<String> valueSerializer = (RedisSerializer<String>) stringRedisTemplate.getValueSerializer();
        byte[] key = keySerializer.serialize("lock:uid1");
        byte[] value = connection.get(key);
        String lid = valueSerializer.deserialize(value);
        if (!lockId.equals(lid)) {
            //lockId不一样,锁被替换。做业务回滚处理
            throw new RuntimeException("锁超时");
        }

        //如果在事务执行之前这key被其他命令所改动,那么下面事务将被打断
        connection.watch(key);
        try {
            //标记一个事务块的开始
            connection.multi();
            //删除key
            connection.del(key);
            //执行所有事务块内的命令
            List<Object> ret = connection.exec();
            if (null == ret) {
                //watch后被其他命令所改动
                throw new RuntimeException("锁超时");
            }
        } finally {
            //取消WATCH命令对key的监视
            connection.unwatch();
        }
        return null;
    }
});

模拟解锁时,持有锁的时间已经超时

stringRedisTemplate.delete("lock:uid1");

String lockId = UUID.randomUUID().toString();
stringRedisTemplate.execute(new RedisCallback<Void>() {
    @Override
    public Void doInRedis(RedisConnection connection) throws DataAccessException {
        RedisSerializer<String> keySerializer = (RedisSerializer<String>) stringRedisTemplate.getKeySerializer();
        RedisSerializer<String> valueSerializer = (RedisSerializer<String>) stringRedisTemplate.getValueSerializer();
        byte[] key = keySerializer.serialize("lock:uid1");
        byte[] value = valueSerializer.serialize(lockId);
        Long ttl = connection.ttl(key, TimeUnit.SECONDS);
        if (-1 == ttl) {
            //清理异常超时
            connection.del(key);
        }
        //setNX将key的值设为value,当且仅当key不存在。
        //若给定的key已经存在,则SETNX不做任何动作。
        Boolean ret = connection.setNX(key, value);
        if (ret) {
            //setNX成功,设置锁的超时时间3秒
            connection.expire(key, 3);
        }
        return null;
    }
});
stringRedisTemplate.execute(new RedisCallback<Void>() {
    @Override
    public Void doInRedis(RedisConnection connection) throws DataAccessException {
        RedisSerializer<String> keySerializer = (RedisSerializer<String>) stringRedisTemplate.getKeySerializer();
        RedisSerializer<String> valueSerializer = (RedisSerializer<String>) stringRedisTemplate.getValueSerializer();
        byte[] key = keySerializer.serialize("lock:uid1");
        byte[] value = connection.get(key);
        String lid = valueSerializer.deserialize(value);
        if (!lockId.equals(lid)) {
            //lockId不一样,锁被替换。做业务回滚处理
            throw new RuntimeException("锁超时");
        }

        //如果在事务执行之前这key被其他命令所改动,那么下面事务将被打断
        connection.watch(key);

        //模拟等待锁3秒后超时
        try {
            Thread.sleep(5 * 1000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        //模拟其他人拿到锁
        stringRedisTemplate.execute(new RedisCallback<Void>() {
            @Override
            public Void doInRedis(RedisConnection connection) throws DataAccessException {
                RedisSerializer<String> keySerializer = (RedisSerializer<String>) stringRedisTemplate.getKeySerializer();
                RedisSerializer<String> valueSerializer = (RedisSerializer<String>) stringRedisTemplate.getValueSerializer();
                byte[] key = keySerializer.serialize("lock:uid1");
                Boolean ret = connection.setNX(key, value);
                if (ret) {
                    //设置锁的超时时间
                    connection.expire(key, 3);
                }
                return null;
            }
        });

        try {
            //标记一个事务块的开始
            connection.multi();
            //删除key
            connection.del(key);
            //执行所有事务块内的命令
            List<Object> ret = connection.exec();
            if (null == ret) {
                //watch后被其他命令所改动
                throw new RuntimeException("锁超时");
            }
        } finally {
            //取消WATCH命令对key的监视
            connection.unwatch();
        }
        return null;
    }
});

完整封装正式可用

1.增加SUBSCRIBE订阅锁释放
2.去掉expire,在value里存放过期时间(使用expire,存在watch失效问题)

    public static class RedisLock {

        private StringRedisTemplate stringRedisTemplate;

        public RedisLock(StringRedisTemplate stringRedisTemplate) {
            this.stringRedisTemplate = stringRedisTemplate;
        }

        public RLock newLock(String redisKey) {
            return new RLock(stringRedisTemplate, redisKey, UUID.randomUUID().toString());
        }

        private static class LockStatus {

            private boolean lock;
            private long ttl;

            public boolean isLock() {
                return lock;
            }

            public long getTtl() {
                return ttl;
            }

            public LockStatus(boolean lock, long ttl) {
                this.lock = lock;
                this.ttl = ttl;
            }
        }

        public static class RLock {

            private static String UNLOCK = "unlock";

            private StringRedisTemplate stringRedisTemplate;
            private String redisKey;
            private String lockId;

            public RLock(StringRedisTemplate stringRedisTemplate, String redisKey, String lockId) {
                this.stringRedisTemplate = stringRedisTemplate;
                this.redisKey = redisKey;
                this.lockId = lockId;
            }

            public void lock(long lockTimeout, TimeUnit unit) {
                stringRedisTemplate.execute(new RedisCallback<Void>() {
                    @Override
                    public Void doInRedis(RedisConnection connection) throws DataAccessException {
                        CompletableFuture future = new CompletableFuture();
                        RedisSerializer<String> keySerializer = (RedisSerializer<String>) stringRedisTemplate.getKeySerializer();
                        byte[] key = keySerializer.serialize(redisKey);
                        //订阅锁释放消息
                        connection.subscribe(new MessageListener() {
                            @Override
                            public void onMessage(Message message, byte[] pattern) {
                                String msg = new String(message.getBody());
                                if (UNLOCK.equals(msg)) {
                                    future.complete(message);
                                }
                            }
                        }, key);
                        //tryLock
                        LockStatus lockStatus;
                        do {
                            lockStatus = _tryLock(lockTimeout, unit);
                            if (!lockStatus.isLock()) {
                                try {
                                    future.get(lockStatus.getTtl(), TimeUnit.MILLISECONDS);
                                } catch (ExecutionException e) {
                                    throw new RuntimeException(e.getMessage(), e);
                                } catch (InterruptedException e) {
                                    Thread.currentThread().interrupt();
                                } catch (TimeoutException e) {
                                }
                            }
                        } while (!lockStatus.isLock());
                        return null;
                    }
                });
            }

            public boolean tryLock(long lockTimeout, TimeUnit unit) {
                LockStatus lockStatus = _tryLock(lockTimeout, unit);
                return lockStatus.isLock();
            }

            public LockStatus _tryLock(long lockTimeout, TimeUnit unit) {
                LockStatus lock = stringRedisTemplate.execute(new RedisCallback<LockStatus>() {
                    @Override
                    public LockStatus doInRedis(RedisConnection connection) throws DataAccessException {
                        RedisSerializer<String> keySerializer = (RedisSerializer<String>) stringRedisTemplate.getKeySerializer();
                        RedisSerializer<String> valueSerializer = (RedisSerializer<String>) stringRedisTemplate.getValueSerializer();
                        byte[] key = keySerializer.serialize(redisKey);
                        long ttl = TimeoutUtils.toMillis(lockTimeout, unit);
                        long expireTime = System.currentTimeMillis() + ttl;
                        byte[] value = valueSerializer.serialize(String.format("%s:%s", lockId, expireTime));
                        String data;
                        do {
                            byte[] bytes = connection.get(key);
                            data = valueSerializer.deserialize(bytes);
                            if (null == data) {
                                //setNX将key的值设为value,当且仅当key不存在。
                                //若给定的key已经存在,则SETNX不做任何动作。
                                Boolean ret = connection.setNX(key, value);
                                if (!ret) {
                                    continue;
                                }
                                //获取锁成功
                                return new LockStatus(true, ttl);
                            }
                        } while (null == data);
                        return new LockStatus(false, getTTL(data));
                    }
                });
                return lock;
            }

            private long getTTL(String data) {
                String[] split = data.split(":");
                Long expireTime = Long.parseLong(split[1]);
                long ttl = expireTime - System.currentTimeMillis();
                return ttl;
            }

            private String getLockId(String data) {
                String[] split = data.split(":");
                String lockId = split[0];
                return lockId;
            }

            public void unlock() {
                stringRedisTemplate.execute(new RedisCallback<Void>() {
                    @Override
                    public Void doInRedis(RedisConnection connection) throws DataAccessException {
                        RedisSerializer<String> keySerializer = (RedisSerializer<String>) stringRedisTemplate.getKeySerializer();
                        RedisSerializer<String> valueSerializer = (RedisSerializer<String>) stringRedisTemplate.getValueSerializer();
                        byte[] key = keySerializer.serialize(redisKey);
                        byte[] value = connection.get(key);
                        String data = valueSerializer.deserialize(value);
                        if (null == data) {
                            //ttl超时。做业务回滚处理
                            throw new RuntimeException("锁超时");
                        }
                        if (getTTL(data) <= 0) {
                            //ttl超时。做业务回滚处理
                            throw new RuntimeException("锁超时");
                        }
                        if (!lockId.equals(getLockId(data))) {
                            //lockId不一样,锁被替换。做业务回滚处理
                            throw new RuntimeException("锁超时");
                        }

                        //如果在事务执行之前这key被其他命令所改动,那么下面事务将被打断
                        connection.watch(key);
                        try {
                            //标记一个事务块的开始
                            connection.multi();
                            //删除key
                            connection.del(key);
                            //执行所有事务块内的命令
                            List<Object> ret = connection.exec();
                            if (null == ret) {
                                //watch后被其他命令所改动
                                throw new RuntimeException("锁超时");
                            }
                            byte[] msg = valueSerializer.serialize(UNLOCK);
                            //发布锁释放消息
                            connection.publish(key, msg);
                        } finally {
                            //取消WATCH命令对key的监视
                            connection.unwatch();
                        }
                        return null;
                    }
                });
            }
        }
    }

测试

    stringRedisTemplate.delete("lock:test");
    RedisLock redisLock = new RedisLock(stringRedisTemplate);
    RedisLock.RLock rLock1 = redisLock.newLock("lock:test");
    rLock1.lock(3 * 1000, TimeUnit.MILLISECONDS);
    rLock1.unlock();
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
下面是一个使用Lua脚本实现Redis分布式锁的代码示例: ```lua -- Lua脚本实现Redis分布式锁 local lockKey = 'lock' local uuid = ARGV\[1\] if redis.call('get', lockKey) == uuid then redis.call('del', lockKey) return 1 else return 0 end ``` 这段代码首先定义了一个锁的键名为`lockKey`,然后通过传入的参数`ARGV\[1\]`获取到要删除的锁的UUID。接下来,它会通过`redis.call('get', lockKey)`来获取当前锁的值,如果与传入的UUID相等,则说明当前锁是由该UUID持有的,此时会使用`redis.call('del', lockKey)`来删除锁,并返回1表示删除成功。如果锁的值与传入的UUID不相等,则说明当前锁不是由该UUID持有的,此时直接返回0表示删除失败。 这段代码可以用于实现Redis分布式锁的原子性删除操作,确保只有持有锁的客户端才能删除锁,避免误删锁的问题。同时,使用Lua脚本可以保证删除锁的操作是原子性的,避免并发情况下的竞争问题。 #### 引用[.reference_title] - *1* *2* [Redis 实现分布式锁+执行lua脚本](https://blog.csdn.net/qq_34285557/article/details/129700808)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^control_2,239^v3^insert_chatgpt"}} ] [.reference_item] - *3* [Redis分布式锁问题(九)Redis + Lua 脚本实现分布式锁](https://blog.csdn.net/weixin_43715214/article/details/127982757)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^control_2,239^v3^insert_chatgpt"}} ] [.reference_item] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值