这两天一直在研究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