如何在 C 函数中保存 Lua 表字段的完整“路径”?
2017-4-8 2:26:58
收藏:0
阅读:86
评论:1
如果我有一个在 Lua 中的全局对象,像这样:
global_object = { }
global_object.stat_group_1 = { }
global_object.stat_group_2 = { }
global_object.stat_group_3 = { }
global_object.stat_group_1.stat_1 = 1 -- 值随时间变化
global_object.stat_group_1.stat_2 = 2 -- 值随时间变化
global_object.stat_group_1.stat_3 = 3 -- 值随时间变化
-- ... 对于其他 stat_groups 也是一样的
我的问题是关于 luaL_ref,lua_rawgeti 和 lua_getfield。我可以使用 luaL_ref 来保存每个 stat 的路径,以避免在堆栈上明确调用它们吗,就像这样:
int global_object_ref;
int stat_group_1_ref;
int stat_1_ref;
// 假设在调用任何 get_* 函数之前已调用了此函数
int start ( lua_State * L )
{
lua_getfield ( L, LUA_RIDX_GLOBALS, "global_object" );
lua_pushvalue ( L, -1 );
global_object_ref = LuaL_ref ( L, LUA_REGISTRYINDEX );
lua_getfield ( L, -1, "stat_group_1" );
lua_pushvalue ( L, -1 );
stat_group_1_ref = LuaL_ref ( L, LUA_REGISTRYINDEX );
lua_getfield ( L, -1, "stat_1" );
lua_pushvalue ( L, -1 );
stat_group_1_ref = LuaL_ref ( L, LUA_REGISTRYINDEX );
return 0;
}
// 这是首选选项。我希望这是可能的
int get_stat1_v1 ( lua_State * L )
{
// stat_1 在 Lua 表中可以有不同的值
lua_rawgeti ( L, LUA_REGISTRYINDEX, stat_1_ref );
// 这是字段 stat_1 的值吗?
int value_of_stat_1 = lua_tointeger ( L, -1 );
return 1;
}
// 这是一种替代方案,如果 v1 不起作用。这行得通吗?
// 再次说明,stat_1 在不同时间可以有不同的值。
int get_stat1_v2 ( lua_State * L )
{
lua_rawgeti ( L, LUA_REGISTRYINDEX, global_object_ref );
lua_rawgeti ( L, LUA_REGISTRYINDEX, stat_group_1_ref );
lua_rawgeti ( L, LUA_REGISTRYINDEX, stat_1_ref );
// 这是字段 stat_1 的值吗?
int value_of_stat_1 = lua_tointeger ( L, -1 );
return 1;
}
请注意,v2 调用了所有保存的引用到堆栈上。那行得通吗?
编辑:在看到 @Nicol Bolas 的回答后,我想提出一个 v3。如果表和子表永远不会被回收,但其值不断更新(想象子表结构是一棵树,每个子表是一个分支,每个基本值是一个叶子。树的结构在执行期间保持不变,但叶子不断更新)。
//这是 v3,我学会了 Lua 注册表与 C 的交互,并提出了一个通过
//直接访问叶子,但间接访问支路的方案。
int get_stat1_v3 ( lua_State * L )
{
// 我可以跳过下面这行直接进入下一行吗?还是我必须遵循整个支路的层次结构?
lua_rawgeti ( L, LUA_REGISTRYINDEX, global_object_ref );// 这有必要吗?
// 我知道我在重复自己,但我想知道上面的调用是否有必要。我能否通过调用此函数直接进入下一步?
lua_rawgeti ( L, LUA_REGISTRYINDEX, stat_group_1_ref );
// 请注意,为了使此操作起作用,我必须从代码中移除对 stat_1 注册表的引用,因为它的实现已经过时。这意味着 stat_1_ref 已被删除。
lua_getfield ( L, -1, "stat_1" );
// 这是字段 stat_1 的实时更新值吗?
int value_of_stat_1 = lua_tointeger ( L, -1 );
return 1;
}
点赞
评论区的留言会收到邮件通知哦~
推荐文章
- Lua 虚拟机加密load(string.dump(function)) 后执行失败问题如何解决
- 我想创建一个 Nginx 规则,禁止访问
- 如何将两个不同的lua文件合成一个 东西有点长 大佬请耐心看完 我是小白研究几天了都没搞定
- 如何在roblox studio中1:1导入真实世界的地形?
- 求解,lua_resume的第二次调用继续执行协程问题。
- 【上海普陀区】内向猫网络招募【Skynet游戏框架Lua后端程序员】
- SF爱好求教:如何用lua实现游戏内调用数据库函数实现账号密码注册?
- Lua实现网站后台开发
- LUA错误显式返回,社区常见的规约是怎么样的
- lua5.3下载库失败
- 请问如何实现文本框内容和某个网页搜索框内容连接,并把网页输出来的结果反馈到另外一个文本框上
- lua lanes多线程使用
- 一个kv数据库
- openresty 有没有比较轻量的 docker 镜像
- 想问一下,有大佬用过luacurl吗
- 在Lua执行过程中使用Load函数出现问题
- 为什么 neovim 里没有显示一些特殊字符?
- Lua比较两个表的值(不考虑键的顺序)
- 有个lua简单的项目,外包,有意者加微信 liuheng600456详谈,最好在成都
- 如何在 Visual Studio 2022 中运行 Lua 代码?

你可以这样做,但你基本上会放弃任何形式的垃圾回收。注册表是 Lua 状态的一部分,因此只要这些表在注册表中,它们必须存在。因此,任何被它们引用的对象都将存在,直到您解除注册或关闭 Lua 状态。
你并不是真正地保存“路径”。你保存的是那些位置的表中实际存储的值。因此,如果表的值更改,存储在注册表中的值不会被更新。保存的
stat_1值将是它当前具有的值,而不是它可能更改的值。