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脚本的校验和