本文共 7948 字,大约阅读时间需要 26 分钟。
一、引言
Redis学了一段时间了,基本的东西都没问题了。从今天开始讲写一些redis和lua脚本的相关的东西,lua这个脚本是一个好东西,可以运行在任何平台上,也可以嵌入到大多数语言当中,来扩展其功能。lua脚本是用C语言写的,体积很小,运行速度很快,并且每次的执行都是作为一个原子事务来执行的,我们可以在其中做很多的事情。由于篇幅很多,一次无法概述全部,这个系列可能要通过多篇文章的形式来写,好了,今天我们进入正题吧。二、Lua简介 Lua 是一个小巧的脚本语言。是巴西里约热内卢天主教大学(Pontifical Catholic University of Rio de Janeiro)里的一个研究小组,由Roberto Ierusalimschy、Waldemar Celes 和 Luiz Henrique de Figueiredo所组成并于1993年开发。 其设计目的是为了嵌入应用程序中,从而为应用程序提供灵活的扩展和定制功能。Lua由标准C编写而成,几乎在所有操作系统和平台上都可以编译,运行。Lua并没有提供强大的库,这是由它的定位决定的。所以Lua不适合作为开发独立应用程序的语言。Lua 有一个同时进行的JIT项目,提供在特定平台上的即时编译功能。 Lua脚本可以很容易的被C/C++ 代码调用,也可以反过来调用C/C++的函数,这使得Lua在应用程序中可以被广泛应用。不仅仅作为扩展脚本,也可以作为普通的配置文件,代替XML,ini等文件格式,并且更容易理解和维护。 Lua由标准C编写而成,代码简洁优美,几乎在所有操作系统和平台上都可以编译,运行。一个完整的Lua解释器不过200k,在目前所有脚本引擎中,Lua的速度是最快的。这一切都决定了Lua是作为嵌入式脚本的最佳选择。三、使用Lua脚本的好处 1、减少网络开销:可以将多个请求通过脚本的形式一次发送,减少网络时延和请求次数。 2、原子性的操作:Redis会将整个脚本作为一个整体执行,中间不会被其他命令插入。因此在编写脚本的过程中无需担心会出现竞态条件,无需使用事务。 3、代码复用:客户端发送的脚步会永久存在redis中,这样,其他客户端可以复用这一脚本来完成相同的逻辑。 4、速度快:见 与其它语言的性能比较, 还有一个 JIT编译器可以显著地提高多数任务的性能; 对于那些仍然对性能不满意的人, 可以把关键部分使用C实现, 然后与其集成, 这样还可以享受其它方面的好处。 5、可以移植:只要是有ANSI C 编译器的平台都可以编译,你可以看到它可以在几乎所有的平台上运行:从 Windows 到Linux,同样Mac平台也没问题, 再到移动平台、游戏主机,甚至浏览器也可以完美使用 (翻译成JavaScript). 6、源码小巧:20000行C代码,可以编译进182K的可执行文件,加载快,运行快。四、redis和lua整合详解 1、调用Lua脚本的语法: $ redis-cli --eval path/to/redis.lua KEYS[1] KEYS[2] , ARGV[1] ARGV[2] ... --eval,告诉redis-cli读取并运行后面的lua脚本 path/to/redis.lua,是lua脚本的位置 KEYS[1] KEYS[2],是要操作的键,可以指定多个,在lua脚本中通过KEYS[1], KEYS[2]获取 ARGV[1] ARGV[2],参数,在lua脚本中通过ARGV[1], ARGV[2]获取。 注意: KEYS和ARGV中间的 ',' 两边的空格,不能省略。redis支持大部分Lua标准库
库名 | 说明 |
---|---|
Base | 提供一些基础函数 |
String | 提供用于字符串操作的函数 |
Table | 提供用于表操作的函数 |
Math | 提供数学计算函数 |
Debug | 提供用于调试的函数 |
redis返回值类型和Lua数据类型转换规则
redis返回值类型 | Lua数据类型 |
---|---|
整数回复 | 数字类型 |
字符串回复 | 字符串类型 |
多行字符串回复 | table类型(数组形式) |
状态回复 | table类型(只有一个ok字段存储状态信息) |
错误回复 | table类型(只有一个err字段存储错误信息) |
Lua数据类型和redis返回值类型转换规则
Lua数据类型 | redis返回值类型 |
---|---|
数字类型 | 整数回复(Lua的数字类型会被自动转换成整数) |
字符串类型 | 字符串回复 |
table类型(数组形式) | 多行字符串回复 |
table类型(只有一个ok字段存储状态信息) | 状态回复 |
table类型(只有一个err字段存储错误信息) | 错误回复 |
script: 是lua脚本
numkeys:表示有几个key,分别是KEYS[1],KEYS[2]...,如果有值,从第numkeys+1个开始就是参数值,ARGV[1],ARGV[2]...
注意: EVAL命令依据参数numkeys来将其后面的所有参数分别存入脚本中KEYS和ARGV两个table类型的全局变量。当脚本不需要任何参数时,也不能省略这个参数(设为0)192.168.127.128:6379>eval "return redis.call('set',KEYS[1],ARGV[1])" 1 name liulei OK 192.168.127.128:6379>get name "liulei"
192.168.127.128:6379>script flush OK 192.168.127.128:6379>SCRIPT FLUSH OK
192.168.127.128:6379>script kill (error)NOTBUSY No scripts in execution right now 192.168.127.128:6379>SCRIPT KILL (error)NOTBUSY No scripts in execution right now //这是当前没有脚本在执行,所以提示该错误
为了防止某个脚本执行时间过长导致Redis无法提供服务(比如陷入死循环),Redis提供了lua-time-limit参数限制脚本的最长运行时间,默认为5秒钟。当脚本运行时间超过这一限制后,Redis将开始接受其他命令但不会执行(以确保脚本的原子性,因为此时脚本并没有被终止),而是会返回“BUSY”错误。
五、安装和使用Lua脚本 1、安装lua类库环境 1.1、yum install -y readline 1.2、yum install -y readline-devel 2、下载lua最新的版本安装 2.1、去官网下载lua,可以直接通过wget下载,地址如下:http://www.lua.org/download.html[root@lunux~]# wget http://www.lua.org/ftp/lua-5.3.4.tar.gz /root/software/download/lua/
2.2、通过ssh SSH Secure File Transfer Client工具,把软件包上传到Linux服务器上。目录是:/root/software/download/lua/
[root@linux~]# cd ./software/download/lua/ [root@linux lua]# tar zxvf lua-5.3.4.tar.gz
2.3、进入到已经解压的目录lua-5.3.4,准备安装文件。
[root@linux lua]# ls [root@linux lua]# lua-5.3.4 lua-5.3.4.tar.gz [root@linux lua]# cd lua-5.3.4 [root@linux lua-5.3.4]#
[root@linux lua-5.3.4]# make linux
[root@linux lua-5.3.4]# make install
2. 6、最后进行测试,进到Linux的命令行,然后输入lua命令,开始测试。
[root@linux lua-5.3.4]# lua >print('lua') lua >print("lua") lua
2.7、按Ctrl+C退出lua命令模式。
>^C [root@linux lua-5.3.4]#
[root@linux lua-5.3.4]# cd /root/application/program/ //执行文件都在这个目录里面 [root@linux program]# mkdir luascript //创建luaScript脚本目录,存放lua脚本文件 [root@linux program]# cd luascript [root@linux luascript]# lua 01.lua //执行01.lua脚本文件
//redis-cli和lua脚本的路径可以是相对路径,也可以是绝对路径 //以下代码就是通过绝对地址来执行 //绝对地址: [root@linux ~]# /root/application/program/redis-tool/redis-cli -h 192.168.127.128 -p 6379 --eval /root/application/program/luascript/02.lua //相对地址: //当前目录 192.168.127.128:6379>pwd [root@linux redis-tool]/root/application/program/redis-tool/ [root@linux redis-tool]# redis-cli -h 192.168.127.128 -p 6379 --eval /root/application/program/luascript/02.lua
//当前redis 数据库中只有name和age两个key,其他数据已经清空。 //当前所在目录 192.168.127.128:6379>keys * 1)"name" 2)"age" 192.168.127.128:6379>get name "liulei" 192.168.127.128:6379>get age "15" //03.lua脚本代码如下: local name=redis.call("get",KEYS[1]) local age=redis.call("get",KEYS[2]) if name=="LLL" then redis.call("set",KEYS[1],ARGV[1]) redis.call("incr",KEYS[2]) end //执行改脚本的命令,必须在Linux的命令行,不是在Redis的命令行 [root@linux ~]# /root/application/program/redis-tool/redis-cli -h 192.168.127.128 -p 6379 --eval /root/application/program/luascript/03.lua name age , patrickLiu //执行脚本命令后 192.168.127.128:6379>keys * 1)"name" 2)"age" 192.168.127.128:6379>get name "patrickLiu" 192.168.127.128:6379>get age "16" //说明带参数的执行Lua脚本成功
//04.lua文件的源码 local b1=redis.call("hgetall",KEYS[1]) return b1 //代码很简单,话不多说 //清空当前数据库 192.168.127.128:6379>flushdb 192.168.127.128:6379>keys * (empty list or set) 192.168.127.128:6379>hmset myhash name zhangsan sex nan address hebeibaoding school laiyuanyizhong OK 192.168.127.128:6379>hmget myhash name sex address school 1)"zhangsan" 2)"nan" 3)"hebeibaoding" 4)"laiyuanyizhong" //我们通过redis客户端获取myhash的结果,进入到redis客户端的当前目录 [root@linux redis-tool]# redis-cli -h 192.168.127.128 -p 6379 --eval ../luascript/04.lua myhash 1)"name" 2)"zhangsan" 3)"sex" 4)"nan" 5)"address" 6)"hebeibaoding" 7)"school" 8)"laiyuanyizhong" //成功获取myhash的列表
转载地址:http://otdgo.baihongyu.com/