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函数

lua 脚本判断是否为空 lua脚本验证_lua

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