Lua环境协作组件
  • 除了创建并修改Lua环境之外,Redis服务器还创建了两个用于与Lua环境进行协作的组件,它们分别是:
    • 负责执行Lua脚本中的Redis命令的伪客户端
    • 用于保存Lua脚本的lua_scripts字典
一、伪客户端
  • 因为执行Redis命令必须有相应的客户端状态,所以为了执行Lua脚本中包含的Redis命令,Redis服务器专门为Lua环境创建了一个伪客户端,并由这个伪客户端负责处理Lua脚本中包含的所有Redis命令
  • Lua脚本使用redis.call函数或者redis.pcall函数执行一个Redis命令,需要完成以下步骤:
    • 1.Lua环境将redis.call函数或者redis.pcall函数想要执行的命令传给伪客户端
    • 2.伪客户端将脚本想要执行的命令传给命令执行器
    • 3.命令执行器执行伪客户端传给它的命令,并将命令的执行结果返回给伪客户端
    • 4.伪客户端接收命令执行器返回的命令结果,并将这个命令结果返回给Lua环境
    • 5.Lua环境在接收到命令结果之后,将该结果返回给redis.call函数或者redis.pcall函数
    • 6.接收到结果的redis.call函数或者redis.pcall函数会将命令结果作为函数返回值返回给 脚本中的调用者
  • 下图展示了Lua脚本在调用redis.call函数时,Lua环境、伪客户端、命令执行器三者之间的通信过程(调用redis.pcall函数时产生的通信过程也是一样的)

Redis(设计与实现):70---Lua脚本之Lua环境协作组件(伪客户端、lua_scripts字典)_伪客户端

  • 举个例子,下图展示了Lua脚本在执行以下命令时,Lua环境、伪客户端、命令执行器三者之间的通信过程

Redis(设计与实现):70---Lua脚本之Lua环境协作组件(伪客户端、lua_scripts字典)_lua_scripts字典_02

Redis(设计与实现):70---Lua脚本之Lua环境协作组件(伪客户端、lua_scripts字典)_lua_03

二、lua_scripts字典
  • 除了伪客户端之外,Redis服务器为Lua环境创建的另一个协作组件是lua_scripts字典:
    • 字典的键为某个Lua脚本的SHA1校验和(checksum)
    • 字典的值则是SHA1校验和对应 的Lua脚本
struct redisServer {
    // ...
    dict *lua_scripts;
    // ...
};
  • Redis服务器会将所有被EVAL命令执行过的Lua脚本,以及所有被SCRIPT LOAD命令载入过的Lua脚本都保存到lua_scripts字典里面
  • lua_scripts字典有两个作用:一个是实现SCRIPT EXISTS命令,另一个是实现脚本复制功能,将在后面的文章介绍
  • 举个例子,如果客户端向服务器发送以下命令:

Redis(设计与实现):70---Lua脚本之Lua环境协作组件(伪客户端、lua_scripts字典)_伪客户端_04

  • 那么服务器的lua_scripts字典将包含被SCRIPT LOAD命令载入的三个Lua脚本,如下图所示:

Redis(设计与实现):70---Lua脚本之Lua环境协作组件(伪客户端、lua_scripts字典)_lua_scripts字典_05