【问题标题】:Lua, metatables, and global variablesLua、元表和全局变量
【发布时间】:2012-04-20 00:40:02
【问题描述】:

我正在努力改进我们在Bitfighter 中为机器人玩家处理 Lua 脚本的方式。目前,每个机器人都有自己的 L 实例,我们正试图通过交换环境表让它们共享一个。请注意,机器人可能是完全不同的脚本。

我意识到这种方法在 Lua 5.2 中已被弃用,但我们目前使用的是 lua-vec,它仍然使用 Lua 5.1。游戏是用 C++ 编写的。

所以...

首先我们创建一个环境,然后调用它:

// Create a table with room for 0 array and 1 non-array elements
lua_createtable(L, 0, 1);                 // -- tab

// Set the globals table to handle any requests that the 
// script's environment can't
lua_pushstring(L, "__index");             // -- tab, "__index"
lua_pushvalue(L, LUA_GLOBALSINDEX);       // -- tab, "__index", _G

// Set table["__index"] = _G, pops top two items from stack
lua_settable(L, -3);                      // -- tab

// Store the new table in the retistry for future use
lua_setfield(L, LUA_REGISTRYINDEX, name); // -- <<empty stack>>

稍后,我们加载一些 Lua 代码,并调用环境表:

luaL_loadfile(L, "luascripts.lua");

lua_getfield(L, LUA_REGISTRYINDEX, name); // -- function, table 
lua_setfenv(L, -2);                       // -- function

然后运行加载的代码:

lua_pcall(L, 0, 0, 0);

当加载的 Lua 尝试使用基本功能时,例如打印,它失败并出现错误:

attempt to call global 'print' (a nil value) 

但是,脚本可以执行以下操作:

__index["print"](12)

那么...为什么我们不能直接访问打印?我们缺少什么?或者有没有更好的方法在同一个 Lua 实例中运行多个脚本?

【问题讨论】:

    标签: c++ lua


    【解决方案1】:

    您的代码接近正确,但包含几个问题 - 您正在尝试做一些不起作用的事情,并且您的尝试以错误的方式做了错误的事情..

    您正在将函数的函数环境设置为如下所示的表:

    {__index = _G}
    

    自然,当您尝试访问print 时,在此表中找不到它。

    从您的 cmets 中,我推断实际上想要设置环境表的 metatable__index 字段。也就是说,您想让环境表类似于下面示例中的t

    t = {}
    setmetatable(t, {__index = _G})
    

    (C++ 的翻译相当简单)。

    不要这样做。它将解决您眼前的问题,但它不会提供足够的沙盒。例如,考虑这样的脚本:

    table.sort = 10
    

    "table" 将由元表事件处理程序在_G 中找到。 sort 只是 table 表的一个元素,因此可以用有罪不罚的方式替换。现在,其他脚本将无法使用table.sort 对表格进行排序。


    执行这种分离的一种方法是通过某种(可能是递归的)用户数据和相关元表事件的手写处理程序来调解对所有全局值的访问。 (这种方式可能具有最大的性能和控制潜力,但可能难以实施)。

    另一种方法是为每个脚本创建一个环境表,并将全局表中的安全/沙盒元素复制到此环境表中(这样每个脚本都有一个完全独立的版本,其中包含全局的所有可变元素)表)。

    很抱歉,我没有时间正确解释我针对您的问题提出的解决方案。我希望我给了你一个开始的地方。随时回来编辑它以包含您最终使用的解决方案。

    【讨论】:

    • 感谢您的回复! “另一种方法是为每个脚本创建一个环境表......”这正是我想要做的,尽管我没有想过为每个脚本提供全局表的副本。我从您的示例中看到了为什么这会提高脚本的安全性和隔离性,但我不确定如何实现这一点,尽管经过大量研究,我也没有找到任何好的实现示例。
    • @Watusimoto:其实很简单。您只需将全局表中的每个列入白名单的元素复制到环境表中(当您获取可变元素时,请确保正确复制元素而不仅仅是引用)。几天后我会回来并举一些适当的例子。 (请提醒我,以免我忘记!)
    • 谢谢。这应该相对简单,因为我的全局表已经仅限于我希望脚本可以访问的函数和值,所以这将是一个批发复制操作。
    猜你喜欢
    • 2021-12-11
    • 1970-01-01
    • 2017-03-05
    • 1970-01-01
    • 2018-06-01
    • 2021-11-07
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多