16、LUA

it2024-04-20  5

  redis服务器创建了一个伪客户端用来执行lua命令,lua_scripts字典用来保存lua脚本

命令:   eval <脚本内容> <key个数> [key列表] [参数列表]:执行指定的脚本   示例:

127.0.0.1:6379> eval 'return "hello " .. KEYS[1] .. ARGV[1]' 1 redis world "hello redisworld"

  evalsha <校验和> <key个数> [key列表] [参数列表]:执行校验和对应的脚本   script exists [sha1]…:判断脚本是否存在   script load <脚本内容>:加载脚本   script flush:删除保存的脚本   script kill:结束正在执行的脚本,如果该脚本执行过写入操作,只能用shtudown nosave命令停止服务器

伪客户端   因为执行redis命令必须要有对应的客户端状态,所以redis服务器专门为lua环境创建了一个伪客户端端用来执行lua脚本命令 执行redis.call和redis.pcall函数过程   1、lua环境将redis.call要执行的脚本命令传送给伪客户端   2、伪客户端将脚本命令发送给命令执行器   3、命令执行器将执行后的结果返回给伪客户端   4、伪客户端将返回的结果返回给lua环境   5、lua环境将返回的结果redis.call函数

redis.call和redis.pcall   区别:redis.call脚本执行错误直接返回,redis.pcall会忽略错误继续执行脚本

lua_scripts字典   用来保存lua脚本,键为脚本的SHA1校验和,值为脚本

eval的实现   1、根据客户端给定的脚本,定义对应的函数(函数名由f_校验和组成)   2、将脚本保存到字典表lua_scripts中   3、执行刚刚定义的函数   示例:

127.0.0.1:6379> eval "return 'helloWorld'" 0 "helloWorld"

    1、计算脚本对应的sha1校验和,"return ‘helloWorld’"脚本对应的校验和是66fa1c42703672f90d1e788f3ee2bf1f2c6cb769     2、在lua环境中定义对应的函数,函数名由f_校验和组成,函数体是脚本本身,如下 function f_66fa1c42703672f90d1e788f3ee2bf1f2c6cb769() return ‘helloWorld’ end     3、执行f_66fa1c42703672f90d1e788f3ee2bf1f2c6cb769函数

执行脚本函数   1、在执行脚本函数前,将eval命令传入的键名参数和脚本参数保存在数组KEYS和ARGV中,并将这两个数组作为全局变量传入到lua环境中。   2、为lua环境装载处理超时钩子(lua-time-limit,默认5秒),这个钩子可以在脚本运行出现错误时,让客户端通过script kill停止脚本或者shutdown nosave关闭服务器。   3、执行函数   4、移除处理钩子   5、将结果保存到输出缓冲区中,等待服务器返回给客户端

evalsha的复制处理 redisServer.repl_scriptcache_dict:保存已经传播给所有从服务器的lua脚本的校验和

最新回复(0)