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 → 释放
相关推荐
不知名的老吴2 分钟前
线程的生命周期之线程同步
java·开发语言·jvm
协享科技5 分钟前
Spring Boot 与 Go 双服务架构实践:从单体拆分到通信设计
java·人工智能·spring boot·后端·架构·golang·ai编程
努力努力再努力wz36 分钟前
【内存管理与高并发内存池系列】从 mmap 到 malloc:文件映射、匿名映射与 glibc 内存分配机制详解
linux·c语言·数据结构·数据库·c++·qt·链表
J2虾虾42 分钟前
C 语言 void 完全用法
c语言·开发语言
码语智行1 小时前
地图上图、空间拓扑查询示例
java·arcgis
程序员黑豆1 小时前
AI全栈开发 - Java:变量
java·前端·ai编程
我是一颗柠檬1 小时前
【Java项目技术亮点】分库分表+数据路由策略:单表5000万后的架构升级方案
java·开发语言·分布式·架构
wu_ye_m1 小时前
学习c语言第35天 函数声明和定义
c语言·开发语言·学习
布朗克1681 小时前
25 IO流高级操作——序列化、NIO与Files工具类
java·数据库·io·nio
小研说技术1 小时前
Spring AI实现rag流程(简易版)
java·后端