Lua C API 中的注册表介绍

原文地址:Lua C API 中的注册表介绍

欢迎参观我的网站:无敌牛 -- 技术/著作/典籍/分享等

在 Lua 的 C API 中,有一个特殊的表,叫做 注册表(Registry) 。它不是 Lua 脚本中能直接访问的普通表,而是专为 C 代码设计的全局存储区


一、注册表是什么?

  • 注册表是一个普通的 Lua 表,但只有 C 代码能访问。
  • 它的"地址"是固定的:使用伪索引 LUA_REGISTRYINDEX(值为 -10000)。
  • Lua 脚本中没有变量指向它,因此脚本无法直接读写它。
  • 它的生命周期和 Lua 状态机(lua_State* L)一致 ------ 只要 L 存在,注册表就存在。
  • 所以,注册表不是 _G(全局表)

二、为什么需要注册表?

当你在 C 中调用 Lua,有时需要:

  • 保存一个 Lua 函数,以后回调它(比如事件处理函数)
  • 保存一个 Lua 表或 userdata,避免被垃圾回收
  • 在多个 C 函数之间传递 Lua 对象

直接保存 lua_State* 和栈位置是不安全的 ------ 栈会变,对象可能被 GC。

注册表提供了一个安全、稳定、长期存储 Lua 值的地方


三、怎么使用注册表?

最常用、最安全的方式是配合 luaL_refluaL_unref

1. 保存值 → 获取引用编号

复制代码
lua_pushvalue(L, idx);        // 把你想保存的值推到栈顶
int ref = luaL_ref(L, LUA_REGISTRYINDEX);  // 存入注册表,返回整数引用

2. 取回值

复制代码
lua_rawgeti(L, LUA_REGISTRYINDEX, ref);  // 把值重新压入栈顶

3. 释放值(重要!)

复制代码
luaL_unref(L, LUA_REGISTRYINDEX, ref);   // 释放引用,避免内存泄漏

四、注意事项

  • 推荐只用 luaL_ref 返回的整数作为键 ------ 这是安全区,不会和 Lua 内部冲突。
  • 避免用字符串键 (如 "mydata")------ Lua 内部可能已占用某些键(如 "IO"),有冲突风险。
  • 用完一定要 luaL_unref ------ 否则对象会一直留在注册表中,造成内存泄漏。

五、典型应用场景

  • 保存 Lua 回调函数供 C 触发
  • 在 C 结构体中存储 Lua 对象的"句柄"
  • 实现 C 与 Lua 之间的对象生命周期管理
  • 跨多个 C 函数共享 Lua 数据

六、总结

注册表是 Lua C API 中一个简单但非常实用的机制:

它让 C 代码能安全地"记住"Lua 中的对象,不怕栈变化,不怕 GC 回收。

使用方法也很简单:

复制代码
luaL_ref → 保存,拿编号
lua_rawgeti → 取回
luaL_unref → 释放
相关推荐
二哈赛车手6 小时前
新人笔记---ApiFox的一些常见使用出错
java·笔记·spring
为何创造硅基生物7 小时前
C语言 结构体内存对齐规则(通俗易懂版)
c语言·开发语言
栗子~~7 小时前
JAVA - 二层缓存设计(本地缓冲+redis缓冲+广播所有本地缓冲失效) demo
java·redis·缓存
YDS8297 小时前
DeepSeek RAG&MCP + Agent智能体项目 —— RAG知识库的搭建和接口实现
java·ai·springboot·agent·rag·deepseek
仰泳之鹅7 小时前
【C语言】自定义数据类型2——联合体与枚举
c语言·开发语言·算法
jolimark8 小时前
C语言自学攻略:小白入门三步走
c语言·编程入门·学习路线·实践项目·自学攻略
未若君雅裁8 小时前
MyBatis 一级缓存、二级缓存与清理机制
java·缓存·mybatis
cen__y9 小时前
Linux12(Git01)
linux·运维·服务器·c语言·开发语言·git
AI人工智能+电脑小能手9 小时前
【大白话说Java面试题 第65题】【JVM篇】第25题:谈谈对 OOM 的认识
java·开发语言·jvm
社交怪人9 小时前
【算平均分】信息学奥赛一本通C语言解法(题号2071)
c语言·开发语言