导出C++中的类到Lua脚本中的经历
这两天一直在研究Lua和LuaPlus。由于项目需要将C++中的类导出到Lua脚本中,让Lua脚本也可以使用C++中的东西。倘若用Lua的C API直接导出到Lua脚本中,感觉不是很方便,所以选择了最新的LuaPlus5.1版,又在网上找了沐枫写的一个LuaPlusHelper.h文件,便开始试着导出C++中的类到Lua脚本中去。我测试用的被导出的类是一个我自己写的,内容相当简单:
1class CExportClass
2{
3public:
4 int GetData() const
5 {
6 return m_Data;
7 }
8
9 void SetData(int Data)
10 {
11 m_Data = Data;
12 }
13protected:
14private:
15 int m_Data;
16};```
17
18
19就为了导出这么一个测试类,我写了如下一个函数:
20
21
22
23```cpp
24void ExportToLua()
25{
26 using namespace LuaPlus;
27 LuaClass<CExportClass>(GLuaState)
28 .create("CExportClass")
29 .def("Get",&CExportClass::GetData)
30 .def("Set",&CExportClass::SetData);
31}```
32
33
34
35
36谁知一编译就报很多如下所示的错误:
37 LuaPlusCD.h(971): error C2780: “int LPCD::Call(RT (__cdecl *)(void),lua_State *,int)” : 应输入 3 个参数,却提供了 4 个LuaPlusCD.h(702) : 参见“LPCD::Call”的声明
38
39
40 为此我查了很久都没能找到问题的根源,只得将沐枫的原例Logger类拿来试一试,与我自己写的类比一比。这一试,我发现在原例中,所有的函数均没有使用const,想必肯定是这个原因,我曾经在自己的项目中也遇到过这样的问题。将const一去掉,OK,编译时以上的错误就永远地消失了!但是问题到此还没有结束,还有一个错误发生在LuaPlusHelper.h头文件的LuaConstructor::ConstructorHelper中,由于我下载的那个LuaPlusHelper.h头文件是用于LuaPlus5.0版本的,而我现在用的是LuaPlus5.1,PushStack函数已经不是LuaObject的成员了,原来的PushStack函数在LuaPlus5.1版本中已经被更名为Push了,于是乎将之改掉.现在编译连接就没有问题了!
41
42
43 编译连接都通过后,我便利用导出的Logger类的来测试一下,还是按照原例中的代码来测试,一运行,没有想到出现问题了:我新建的一个Logger对象,在使用其成员函数时的this指针却不是生成时的this指针,也就是说脚本中找到的我需要的对象的地址有问题!我将此测试放到LuaPlus5.0版本中,测试也会遇到同样的问题.经过跟踪调试,发现原来是LPCD::GetObjectUserData函数有问题,原代码:
44 if (type == LUA_TUSERDATA)
45 return lua_unboxpointer(L, 1);
46
47
48 其中lua_unboxpointer是一个宏,其定义为#define lua_unboxpointer(L,i) (*(void **)(lua_touserdata(L, i)))
49 跟踪调试时发现lua_touserdata函数本身返回的就是我们需要的对象的正确地址,但是经过(*(void **)这么一转换就变成了另一个地址了.将之改为:
50 if (type == LUA_TUSERDATA)
51 return lua_touserdata(L, 1);
52 现在编译运行,OK!全部正确!至此,整个C++成员函数的导出成功!
53
- 原文作者:Witton
- 原文链接:https://wittonbell.github.io/posts/2007/2007-05-18-导出C++中的类到Lua脚本中的经历/
- 版权声明:本作品采用知识共享署名-非商业性使用-禁止演绎 4.0 国际许可协议. 进行许可,非商业转载请注明出处(作者,原文链接),商业转载请联系作者获得授权。