当前位置:首页 > 技术 > 正文内容

Lua 虚拟机指令解析:函数操作与闭包实现

访客 技术 2026年7月6日 1

函数嵌套与 FuncState 链式结构

在 Lua 编译阶段,解析器通过 FuncState 结构来维护当前编译函数的上下文。当存在函数嵌套时,FuncState 通过 prev 指针形成一条链表,指向父级编译器的状态。以下示例展示了这一结构:

-- 全局层 (root_fs: prev=NULL)
local master_var = 100

function outer_task() -- (outer_fs: prev=root_fs)
    local scope_var = 2
    local function inner_task() -- (inner_fs: prev=outer_fs)
        local hidden_var = 5
        print(hidden_var, scope_var, master_var)
    end
    inner_task()
end

在该模型中,inner_task 的闭包需要访问父作用域的变量,这涉及到 UpValue 的捕获机制。

函数解析流程

当解析器遇到 function 关键字时,会触发嵌套解析流程。以简单的空函数为例:

function sample() end

其生成的 Lua 字节码核心指令如下:

[1] closure   0  0   ; 实例化闭包原型
[2] setglobal 0  0   ; 将闭包对象绑定至全局变量 sample
[3] return    0  1   ; 函数返回

OP_CLOSURE 指令在指令集中的定义如下:

/* 
 * A: 存放闭包对象的寄存器索引
 * Bx: 存放在 Proto 数组中的原型索引
 */
OP_CLOSURE, 

代码构建核心路径

Lua 解析器遵循递归下降的策略。在语法分析过程中,body 函数负责生成闭包的 Proto 结构,并将其挂载到父级作用域的存储数组中:

static void compile_body(LexState *ls, expdesc *result, int is_method, int line) {
  FuncState child_fs;
  open_func(ls, &child_fs); // 初始化子作用域

  // 1. 解析参数表与函数体
  parse_parameters(ls);
  parse_block(ls);

  // 2. 完成闭包原型构建
  close_func(ls);
  
  // 3. 将闭包指针放入父级的 Proto 列表
  attach_closure(ls, &child_fs, result);
}

闭包与变量绑定

当函数体解析完成后,attach_closure(对应原逻辑中的 pushclosure)将生成的 Proto 指针提交到父级 FuncState

static void attach_closure(LexState *ls, FuncState *child, expdesc *desc) {
  FuncState *parent = ls->fs;
  Proto *p = parent->f;
  
  // 扩展原型列表空间
  expand_proto_list(ls, p);
  p->sub_protos[parent->sub_count++] = child->f;
  
  // 生成 OP_CLOSURE 指令
  init_exp(desc, VRELOCABLE, luaK_codeABx(parent, OP_CLOSURE, 0, parent->sub_count - 1));
}

运行时闭包结构 (Closure)

Lua 中的 Closure 在运行时分为 C 闭包(CClosure)和 Lua 闭包(LClosure)。对于用户定义的函数,采用 LClosure 结构:

typedef struct LClosure {
  ClosureHeader;
  struct Proto *p;      // 函数逻辑的编译原型
  UpVal *upvals[1];     // 闭包捕获的外部变量引用
} LClosure;

变量存储分配

函数定义完成后的赋值操作(如 setglobalsetupval)由 luaK_storevar 处理。该函数根据变量作用域类型执行不同的指令生成逻辑:

void luaK_storevar(FuncState *fs, expdesc *var, expdesc *val) {
  switch (var->kind) {
    case VLOCAL:
      emit_move(fs, var, val);
      break;
    case VUPVAL:
      emit_opcode(fs, OP_SETUPVAL, val, var->info);
      break;
    case VGLOBAL:
      emit_opcode(fs, OP_SETGLOBAL, val, var->info);
      break;
    case VINDEXED:
      emit_opcode(fs, OP_SETTABLE, var, val);
      break;
  }
}

相关文章

Linux crontab 详解

1) crontab 是什么cron 是 Linux 的定时任务守护进程;crontab 是用来编辑/查看“按时间周期执行命令”的表(cron table)。常见两类:用户 crontab:每个用户一份(crontab -e 编辑)系统级 crontab / cron.d:可指定执行用户(/etc/crontab、/etc/cron.d/*)2) crontab 时间...

富文本里可以允许的 HTML 属性

一、所有标签默认允许的安全属性(极少)class        (可选)id           (通常建议禁用)title️ 注意:id 容易被滥用做锚点注入,很多系统直接禁用class 允许的话最好只允许固定前缀(如 editor-*)二、a 标签允许属性<a href="" t...

Mac 安装 Node.js 指南

方法一:通过官网安装包(最简单,适合初学者)如果你只是想快速安装并开始使用,这是最直接的方法。访问 Node.js 官网。页面会显示两个版本:LTS (Recommended For Most Users):长期支持版,最稳定。建议选这个。Current:最新特性版,包含最新功能但可能不够稳定。下载 .pkg 安装包并运行。按照安装向导点击“下一步”即可完成。方法二:使用 Homebrew 安装(...

Dom\HTML_NO_DEFAULT_NS 的副作用:自动加闭合标签

在使用Dom\HTMLDocument时,Dom\HTML_NO_DEFAULT_NS 将禁止在解析过程中设置元素的命名空间, 此设置是为了与DOMDocument向后兼容而存在的。当使用它时,已知的一个副作用就是:自动加闭合标签例如 </img> 为什么会这样?当你使用:Dom\HTML_NO_DEFAULT_NS文档会变成 无命名空间模式,此时内部更接近 XML...

Laravel 事件和监听器创建

在 Laravel 中,使用 Artisan 命令创建 Events(事件) 和 Listeners(监听器) 是非常高效的。你可以通过以下几种方式来实现:1. 手动创建单个 Event如果你只想创建一个事件类,可以使用 make:event 命令:Bashphp artisan make:event UserRegistered执行后,文件将生成在 app/Even...

自定义域名解析神器 dnsmasq

什么是 dnsmasq?dnsmasq 是一个轻量级、功能强大的网络服务工具,专为小型和中等规模网络设计。它是一个综合的网络基础设施解决方案[1]。dnsmasq 能做什么?功能说明应用场景DNS 转发与缓存将 DNS 查询转发到上游服务器(ISP、Google DNS 等),并在本地缓存结果加快 DNS 查询速度,减少外部 DNS 流量本地 DNS解析本地网络设备的主机名,无需编辑&n...

发表评论

访客

◎欢迎参与讨论,请在这里发表您的看法和观点。