如何在 C 函数中保存 Lua 表字段的完整“路径”?

如果我有一个在 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_reflua_rawgetilua_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;
}
点赞
用户734069
用户734069

你可以这样做,但你基本上会放弃任何形式的垃圾回收。注册表是 Lua 状态的一部分,因此只要这些表在注册表中,它们必须存在。因此,任何被它们引用的对象都将存在,直到您解除注册或关闭 Lua 状态。

你并不是真正地保存“路径”。你保存的是那些位置的表中实际存储的值。因此,如果表的值更改,存储在注册表中的值不会被更新。保存的 stat_1 值将是它当前具有的值,而不是它可能更改的值。

2017-04-08 01:00:55