LUA脚本

目录

一、Lua脚本

Lua脚本中可以编写多条Redis命令,确保多条命令共同执行时是一个原子操作

1.编写脚本:

Lua执行redis命令的语法:redis.call('命令名称', 'key', '其它参数',...)

lua 复制代码
-- 释放锁的lua脚本
-- 锁的key
local key = KEYS[1]
-- 当前线程标识
local threadId = ARGV[1]

-- 获取锁中的线程标识
local id = redis.call('get',key)
-- 比较线程标识与锁中的标识是否一致
if(id == threadId) then
    -- 释放锁
    return redis.call('del', key)
end

return 0

2.调用脚本:

语法:EVAL 脚本内容字符串 参数数量 key [key...] arg [arg...]

脚本中的key、value可以作为参数传递,key类型参数会放入KEYS数组,其它参数会放入ARGV数组,使用数组下标取值,默认下标从1开始

java 复制代码
// 不加参数(无形参)
EVAL "return redis.call('set', 'name', 'jack')" 0
// 加参数
EVAL "return redis.call('set', KEYS[1], ARGV[1])" 1 name jack

StringRedisTemplate调用Lua脚本的API如下:

java 复制代码
// 调用lua脚本释放锁
public void unlock1(String name) {
    // RedisScript是一个脚本类,用来接收lua脚本文件
    DefaultRedisScript<Long> unlock_script = new DefaultRedisScript<>();
    // ClassPath默认是resources文件夹,设置lua脚本文件位置
    unlock_script.setLocation(new ClassPathResource("unlock.lua"));
    // 设置lua脚本的返回值
    unlock_script.setResultType(Long.class);

    // 调用lua脚本,注意KEYS传的是List集合
    stringRedisTemplate.execute(unlock_script, Collections.singletonList("lock:" + name), String.valueOf(Thread.currentThread().getId()));
}

3.Lua实现可重入锁:

3.1 获取锁:

lua 复制代码
local key = KEYS[1]; --锁的key,一人一单问题就是用户id,同一用户的多个线程竞争同一把锁
local threadId = ARGV[1]; --持有锁的线程标识
local releaseTime = ARGV[2]; --锁的有效期

--锁是未被占有
if(redis.call('exists', key)==0) then
    redis.call('hset', key, theadId, '1'); --获取锁
    redis.call('expire', key, releaseTime); --设置有效期
    return 1; --返回表示获取成功
end;

--锁已被占有,且锁是本线程占有的
if(redis.call('hexists', key, threadId)==1) then
    redis.call('hincrby', key, threadId, '1'); --重入计数自增1
    redis.call('expire', key, releaseTime); --重置有效期
    return 1;
end;

--锁已被占有,且不是本线程占有的
return 0;

3.2 释放锁:

lua 复制代码
local key = KEYS[1]; --锁的key,一人一单问题就是用户id,同一用户的多个线程竞争同一把锁
local threadId = ARGV[1]; --持有锁的线程标识
local releaseTime = ARGV[2]; --锁的有效期

--当前锁不是被本线程持有
if(redis.call('hexists', key, threadId)==0) then
    return nil;
end;

--当前所被本线程持有
local count = redis.call('hincrby', key, threadId, -1); --可重入次数-1
--可重入次数是否减为0
if(count>0) then
    redis.call('expire', key, releaseTime) --重置有效期
    return nil;
else
    redis.call('del', key)
end;

return nil;    
相关推荐
今夕资源网1 分钟前
powershell工具包 安装升级脚本并设置UTF-8 环境快捷方式创建 将powershell的编码默认改为UTF-8
开发语言·utf-8·powershell·utf-8编码·powershell7·powershell5·设置utf-8编码
机器视觉知识推荐、就业指导27 分钟前
Qt:真正的门槛不是入门,而是维护
开发语言·qt
hhb_6181 小时前
Dylan 语言核心特性与工程实践深度解析
开发语言·c#
无巧不成书02181 小时前
零基础Java网络编程全解:从核心概念到Socket实战,一文打通Java网络通信
java·开发语言·网络
饭小猿人1 小时前
Flutter实现底部动画弹窗有两种方式
开发语言·前端·flutter
aq55356002 小时前
Workstation神技:一键克隆调试环境
java·开发语言
lly2024062 小时前
框架:构建高效系统的基石
开发语言
skywalk81632 小时前
发现Kotti项目的python包Beaker 存在安全漏洞
开发语言·网络·python·安全
天天进步20153 小时前
Python全栈项目:从零构建基于 Django 的知识管理系统(KMS)
开发语言·python·django
珎珎啊3 小时前
Python3 迭代器与生成器
开发语言·python