Redis中的Lua脚本怎么玩(redis的lua脚本有什么用)
haoteby 2025-05-08 00:57 9 浏览
Redis中的Lua脚本怎么玩
Lua是一门强大、快速、轻量的嵌入式脚本语言,我们日常开发中接触的最多的还是Redis为保证原子性使用Lua执行多命令的一种方法,那么现在先来熟悉Lua基本用法。
Lua安装
Lua现在最新版本5.4.4,官网地址https://www.lua.org/
-- 解压
tar zxf lua-5.4.4.tar.gz
-- 进入lua解压文件主目录
cd lua-5.4.4
-- 编译
make all test
Lua官网最新是5.4的,如果想参考中文文档可以按照5.3的版本来,中文翻译地址https://www.runoob.com/manual/lua53doc/
Lua的基本用法
Lua的基本命令都可以在Lua官方提供的测试地址进行命令测试https://www.lua.org/cgi-bin/demo
Lua申明类型
Lua中分为全局变量和局部变量,一般建议将变量定义为布局变量,效率更高
--- 全局变量
name = 'felord.cn'
--- 局部变量
local age = 18
Lua数据类型
Lua中包含的数据类型有8种,string、boolean、nil、table、number、userdata、function、thread,如果仅在Redis中使用那么function、thread、userdata不建议使用,只使用其余五种就行。
这五种中,最难理解的应该是table,table和Java中提到的Hash类型有点类似,但并不是完全相同。
arr = {'zhangsan','lisi',1,true,2.1}
print(arr[1]) -- table下标从1开始
print(arr[3])
print(arr[6])
print(#arr) -- 获取table的长度
---------- 结果
zhangsan
1
nil
5
采用字典模式时,用法不一样
--- 定义字典类型
arr = {name='zhangsan',age=12,sex='男'}
print(arr[1]) -- 普通取值无效
print(arr['name'])
print(#arr) -- 常规获取table长度失效
----------- 结果
nil
zhangsan
0
采用混合模式时
-- 采用混合模式定义table
arr = {name='zhangsan',age=12,1,sex='男',2.2}
print(arr[3])-- 字典类型不会统计所以arr[**]这种格式只能针对普通值
print(arr[1])--- 能够查询
print(arr[2])--- 能够查询
print(#arr)--- 只能查询普通值
print(arr['age'])
print(arr['sex'])
-----------结果
nil
1
2.2
2
12
男
在计算table的长度时,不能仅仅通过#arr获取,因为可能存在混合模式的情况,在不清楚元素类型的情况下建议采用循环获取table长度
Lua判断语句
-- 定义局部变量
local a = 21
if a < 10 then
print('a<10')
elseif a < 20 then
print('20<a<=10')
else
print('a>=20')
end
Lua循环判断
local arr = {22,23,44,name='zhangsan'}
for i,v in ipairs(arr) do
print('i = '..i)-- 下标
print('v = '..v)-- 值
end
-- 结果 只能循环普通值,不能循环字典属性
i = 1
v = 22
i = 2
v = 23
i = 3
v = 44
通用循环可以循环字典属性
local arr = {22,23,44,name='zhangsan'}
-- pairs通用循环可以循环字典属性,ipairs只能循环普通值
for i,v in pairs(arr) do
print('i = '..i)
print('v = '..v)
end
--- 结果
i = 1
v = 22
i = 2
v = 23
i = 3
v = 44
i = name
v = zhangsan
Redis中Lua使用
Redis从2.6.0版本开始支持Lua脚本,在Redis中使用不需要另外安装Lua程序,Redis内嵌了Lua。
EVAL
eval定义Redis执行的命令,格式为EVAL script numkeys key [key ...] arg [arg ...]
-- 在EVAL的script中添加Lua脚本时,keys代表键值,argv代表value值是全局运行变量不能写错
-- numkeys代表键的个数,是必须的参数,不能写错
127.0.0.1:6379> EVAL "return redis.call('set',KEYS[1],ARGV[1])" 1 name zhagnsan
OK
127.0.0.1:6379> keys *
1) "name"
127.0.0.1:6379> get name
"zhagnsan"
127.0.0.1:6379> EVAL "return redis.call('get',KEYS[1])" 1 name
"zhagnsan"
让redis执行lua脚本有两种形式,call和pcall
-- call正常返回错误信息,不做任何处理
127.0.0.1:6379> EVAL "return redis.call('no_command')" 0
(error) ERR Error running script (call to f_1e6efd00ab50dd564a9f13e5775e27b966c2141e):
@user_script:1: @user_script: 1: Unknown Redis command called from Lua script
-- pcall返回处理好的错误信息
127.0.0.1:6379> EVAL "return redis.pcall('no_command')" 0
(error) @user_script: 1: Unknown Redis command called from Lua script
注意点
在Redis中使用Lua时需要注意两点,Redis官方也提到这两点
精度丢失问题
在Redis中执行Lua脚本,因为是两个不同的编程环境,Lua脚本并不是区分整数和浮点数,所以Lua脚本将Lua数值转换为Redis中的值时会丢失精度,其它类型相互转换可以参考官网
-- 在将Lua中的值返回Redis
127.0.0.1:6379> EVAL "return {1,4,3.33,'zhangsan'}" 0
1) (integer) 1
2) (integer) 4
3) (integer) 3 -- 精度丢失
4) "zhangsan"
-- 可以之间返回字符串,这样精度不会丢失
127.0.0.1:6379> EVAL "return {1,4,'3.33','zhangsan'}" 0
1) (integer) 1
2) (integer) 4
3) "3.33"
4) "zhangsan"
-- 或者使用字符串转换函数
127.0.0.1:6379> EVAL "return {1,4,tostring(3.33),'zhangsan'}" 0
1) (integer) 1
2) (integer) 4
3) "3.33"
4) "zhangsan"
nil值处理
在前面提到Lua的8大基本数据类型就包含了nil这个类型,这个类型是Lua所特有的,无法转换到Redis中,如果在转换时遇到了nil这个值,这样将导致转换停止
127.0.0.1:6379> EVAL "return {1,4,'3.33',nil,'zhangsan'}" 0
1) (integer) 1
2) (integer) 4
3) "3.33" -- 本来后面应该还有zhangsan这个数据,但是遇到了nil转换直接停止了
Lua脚本
Lua脚本的执行都是原子性的,所以Lua脚本执行势必阻塞其它线程,那么Lua脚本不宜过大,过大会带来其它资源的消耗,也不宜将一些复杂逻辑放入Lua脚本中。
SCRIPT LOAD
Lua脚本的预加载,避免多次传输,可以重复使用
127.0.0.1:6379> SCRIPT LOAD "return 'hello lua'"
"aeebf56de5e46b1d6f9e154df45402368df5a1d8" -- 得到唯一字符串表示
-- 配合EVALSHA使用,和EVAL类似可以传递传参
127.0.0.1:6379> EVALSHA aeebf56de5e46b1d6f9e154df45402368df5a1d8 0
"hello lua"
SCRIPT EXISTS
检查脚本是否存在
-- 存在返回1,不存在返回0
127.0.0.1:6379> SCRIPT EXISTS aeebf56de5e46b1d6f9e154df45402368df5a1d8
1) (integer) 1
SCRIPT KILL
终止正在执行的脚本,如果当终止脚本执行写逻辑还未结束,这时SCRIPT KILL命令是无效的,因为这违反了Lua的原子性规则,这种情况可以使用SHUTDOWN NOSAVE命令强制结束
SCRIPT FLUSH
清空脚本缓存
SCRIPT DEBUG
如果测试bug有误还可以开启debug模式
127.0.0.1:6379> SCRIPT DEBUG yes
OK
127.0.0.1:6379> SCRIPT help
2) DEBUG (YES|SYNC|NO)
3) Set the debug mode for subsequent scripts executed.
相关推荐
- 网站seo该怎么优化
-
一、网站定位在建设一个网站之前,我们首先要做的就是一个网站清晰的定位,会带来转化率相对较高的客户群体,我们建站的目的就是为了营销,只有集中来做某一件事,才会更好的展现我们的网站。在做SEO优化的同时...
- 3个小技巧教你如何做好SEO优化
-
想半路出家做SEO?可是,怎么才做的好呢?关于SEO专业技术弄懂搜索引擎原理,咱们做搜索引擎排名的首先就是要了解搜索引擎的工作原理,对SEO优化有更深入了解之后再来做SEO,你就能从搜索引擎的视点...
- SEO指令分享:filetype指令
-
filetype用于搜索特定的文件格式。百度和谷歌都支持filetype指令。比如搜索filetype:pdf今日头条返回的就是包含今日头条这个关键词的所有pdf文件,如下图:百度只支持:pdf...
- 网站seo优化技巧大全
-
SEO在搜索引擎中对检索结果进行排序,看谁最初是在用户的第一眼中看到的。实际上,这些排名都是通过引擎的内部算法来实现的。例如,百度算法很有名。那么,对百度SEO的优化有哪些小技巧?下面小编就会说下针对...
- 小技巧#10 某些高级的搜索技巧
-
由于某些原因,我的实验场所仅限百度。1.关键词+空格严格说来这个不能算高级,但关键词之间打空格的办法确实好用。我习惯用右手大拇指外侧敲击空格键,这个习惯在打英文报告时尤其频繁。2.site:(请不要忽...
- MYSQL数据库权限与安全
-
权限与安全数据库的权限和数据库的安全是息息相关的,不当的权限设置可能会导致各种各样的安全隐患,操作系统的某些设置也会对MySQL的安全造成影响。1、权限系统的工作原理...
- WPF样式
-
UniformGrid容器<UniformGridColumns="3"Rows="3"><Button/>...
- MySQL学到什么程度?才有可以在简历上写精通
-
前言如今互联网行业用的最多就是MySQL,然而对于高级Web面试者,尤其对于寻找30k下工作的求职者,很多MySQL相关知识点基本都会涉及,如果面试中,你的相关知识答的模糊和不切要点,基...
- jquery的事件名称和命名空间的方法
-
我们先看一些代码:当然,我们也可以用bind进行事件绑定。我们看到上面的代码,我们可以在事件后面,以点号,加我们的名字,就是事件命名空间。所谓事件命名空间,就是事件类型后面以点语法附加一个别名,以便引...
- c#,委托与事件,发布订阅模型,观察者模式
-
什么是事件?事件(Event)基本上说是一个用户操作,如按键、点击、鼠标移动等等,或者是一些提示信息,如系统生成的通知。应用程序需要在事件发生时响应事件。通过委托使用事件事件在类中声明且生成,且通过...
- 前端分享-原生Popover已经支持
-
传统网页弹窗开发需要自己处理z-index层级冲突、编写点击外部关闭的逻辑、管理多个弹窗的堆叠顺序。核心优势对比:...
- Axure 8.0 综合帖——新增细节内容
-
一、钢笔工具与PS或者AI中的钢笔工具一样的用法。同样有手柄和锚点,如果终点和起点没有接合在一起,只要双击鼠标左键即可完成绘画。画出来的是矢量图,可以理解为新的元件。不建议通过这个工具来画ICON图等...
- PostgreSQL技术内幕28:触发器实现原理
-
0.简介在PostgreSQL(简称PG)数据库中,触发器(Trigger)能够在特定的数据库数据变化事件(如插入、更新、删除等)或数据库事件(DDL)发生时自动执行预定义的操作。触发器的实现原理涉及...
- UWP开发入门(十七)--判断设备类型及响应VirtualKey
-
蜀黍我做的工作跟IM软件有关,UWP同时会跑在电脑和手机上。电脑和手机的使用习惯不尽一致,通常我倾向于根据窗口尺寸来进行布局的变化,但是特定的操作习惯是依赖于设备类型,而不是屏幕尺寸的,比如聊天窗口的...