你如何通过Lua C API将C++成员方法和成员变量绑定在一起?

到目前为止,我所做的所有搜索都找到了非常接近但不完全符合我尝试做的事情的东西。

让我用最基本的方式描述一下:

  • 想象你有一个 C++ 类
class A
{
public:
    int Method();
    int Variable;
};
  • 现在想象你实例化了 A* Foo;

  • 现在想象你有一个包含这个 3 行函数的 .lua 文件:

function Test()
    local n = Foo:Method();
    Foo.Variable = 0;
    local m = Foo.Variable;
end

如何将对象 A* 绑定到 Lua,以便所有这些操作都是可行的?

伪代码方面,我的第一次尝试是这样的,部分参考了复制粘贴的示例:

  1. 在一个仅被调用一次的函数中,无论 A 的实例数量为多少:
    • 创建 newmetatable( MT )
    • pushvalue( -1 )(我不太明白这个)
    • setfield( -2, "__index" )
    • pushcfunction,使用从 checkudata 中拆开的 A* 指针的静态版本
    • setfield( -2, "Method" )
  2. 在每个实例都会调用的 init 函数中,例如 Foo:
    • 使用 newuserdata 创建指向 Foo 的指针
    • setmetatable( MT )
    • setglobal(Foo),以便名称可供 Lua 使用
  3. 在主要的测试函数中的 pcall 函数中:
    • 使用 .lua 中提到的 3 行调用测试函数

在这样做时,Foo:Hide(); 成功调用了我的静态函数,它成功地拆开了指针并调用了其成员 Hide()。

到目前为止,:Method() 运行良好。

然后我尝试支持 .Variable 访问。每个人似乎都在说再次使用元表,这次覆盖 __index 和 __newindex 并使它们成为一种通用的 Get/Set,其中你支持某些键作为有效的变量链接,例如 if( key == "Variable" ) Variable = val;

这也很好用。

问题在于尝试将这两个东西结合在一起。一旦你使用一个 getter/setter 覆盖了 __index/__newindex 以处理 Variable,Method() 调用就不再调用 Method() 静态函数,而是进入了你所绑定的 __index 函数中。

所有这些说法都非常好,但如何支持这些看似基本的用例组合呢?

实际的代码片段比纯理论更受欢迎。

提醒:请仅使用基本的 C API 进行回复,不要使用第三方库。

点赞
用户298661
用户298661

Method() 调用不再调用 Method() 静态函数,而是进入你绑定的 __index 函数。

因此,编写程序,如果键存在于表中,则首先返回该键,否则将使用 getter/setter。

2014-05-22 11:32:23