【Lua】Lua脚本及在Redis中的使用

一.Lua脚本介绍

  Lua 是用标准C语言编写并以源代码形式开放的一种轻量小巧的脚本语言,设计目的是为了嵌入应用程序中,为应用程序提供灵活的扩展和定制功能。由于体积小只有200多kb,可以很容易放入集成在一些软件系统里。可以为一些中间件提供支持功能,比如nginx,redis。 Lua脚本可以很容易的被C/C++ 代码调用,也可以反过来调用C/C++的函数。

  Lua官网 :http://www.lua.org/

  Lua官方文档:http://www.lua.org/manual/5.3/

  Lua安装手册:https://www.php.cn/lua/lua-environment.html

 

二.Lua脚本安装

  安装lua

curl -R -O http://www.lua.org/ftp/lua-5.3.0.tar.gz
tar zxf lua-5.3.0.tar.gz
cd lua-5.3.0
make linux test
make install

 

如果在make linux test 这步出现问题    lua.c:80:31: 致命错误:readline/readline.h:没有那个文件或目录
执行以下语句
解决:
yum  install   libreadline-dev  这个我本地不行
yum -y install readline-devel

或者
yum install readline-dev readline-devel

安装完成后 然后在执行
make linux test
make install

安装完成后 查看是否成功安装

说明已经lua已经安装成功

在查看版本

在创建一个脚本试下lua一个程序

创建一个 脚本HelloWorld.lua

里面编写

print("lua Hello World!")

然后执行

 

 三.Lua脚本和Redis关系

  Reids 2.6 版本中自带Lua 脚本功能, 通过内嵌对 Lua 环境的支持, Redis 解决了长久以来不能高效地处理 CAS (check-and-set)命令的缺点,比如使用Redis原生命令,需要从Redis中获取某个key,然后提取其中的值进行比对,如果相等就不做处理,如果不相等或者key不存在则将key设置成目标值。仅仅一个单点的compare and set操作就需要与Redis通讯两次。此外,这种分散操作无法利用Redis的原子特性,占用多次网络IO。使用Lua脚本就可以通过组合打包使用多个命令, 轻松实现以前很难实现或者不能高效实现的模式。

 

四.Redis中使用Lua脚本

  redis中通过EVAL命令去调用Lua。

    官网说明地址: http://www.redis.cn/commands/eval.html

    语法:

EVAL script numkeys key [key ...] arg [arg ...] 

 

参数说明:

  • script:     参数是一段 Lua 脚本程序,是一段lua脚本字符串。
  • numkeys: 用于指定键名参数的个数。
  • key [key ...]: 从 EVAL 的第三个参数开始算起,表示在脚本中所用到的那些 Redis 键(key),这些键名参数可以在 Lua 中通过全局变量 KEYS 数组,用 1 为基址的形式访问( KEYS[1] , KEYS[2] ,以此类推)。
  • arg [arg ...]: 附加参数,在 Lua 中通过全局变量 ARGV 数组访问,访问的形式和 KEYS 变量类似( ARGV[1] 、 ARGV[2] )。

 

具体调用示例

127.0.0.1:6379> EVAL "redis.call('SET','redis-call-lua-key','redis-call-value')" 0      #redis.call是redis搭建在lua环境中默认加载的类库才有的
(nil)
127.0.0.1:6379> get redis-call-lua-key      #查看执行成功
"redis-call-value"

 上面返回nil是因为lua脚本没有返回值,redis接受不到返回值就返回nil。加上return就可以返回ok。

127.0.0.1:6379> EVAL "return redis.call('SET','redis-call-lua-key','redis-call-value1')" 0
OK
127.0.0.1:6379> get redis-call-lua-key
"redis-call-value1"

 

我们也可以在redis中缓存Lua脚本,然后下次使用时候在调用。

先看看如何缓存Lua脚本

127.0.0.1:6379> SCRIPT LOAD "return redis.call('SET',KEYS[1],ARGV[1])"
"cf63a54c34e159e75e5a3fe4794bb2ea636ee005"        

 执行缓存的Lua脚本

127.0.0.1:6379> EVALSHA cf63a54c34e159e75e5a3fe4794bb2ea636ee005 2 LuaKey1 LuaKey2 LuaValue
OK
127.0.0.1:6379> get LuaKey1
"LuaValue"

  通过SCRIPT LOAD命令将Lua脚本命令缓存,并生成返回了一个固定长度的hash字符串,不管你的Lua脚本命令由多长,都会返回这个定长的hash字符。这样做的好处是减少网络的传输,因为redis客户端向服务器传输大段的lua脚本命令时候,这回增加网络开销,而将Lua命令缓存在redis中,客户端只需要传递这个固定长度的hash字符串来减少传输的消耗,同时脚本缓存在redis中也可以防止脚本被篡改。

 





posted @ 2018-02-07 15:38  songguojun  阅读(1445)  评论(0编辑  收藏  举报