导出C++中的类到Lua脚本中的经历收藏

来源:互联网 发布:网约车软件开发 编辑:程序博客网 时间:2024/04/30 00:22
导出C++中的类到Lua脚本中的经历收藏

新一篇: C++类模板的成员函数模板写法 | 旧一篇: VSLua工具

        这两天一直在研究Lua和LuaPlus。由于项目需要将C++中的类导出到Lua脚本中,让Lua脚本也可以使用C++中的东西。倘若用Lua的CAPI直接导出到Lua脚本中,感觉不是很方便,所以选择了最新的LuaPlus5.1版,又在网上找了沐枫写的一个LuaPlusHelper.h文件,便开始试着导出C++中的类到Lua脚本中去。我测试用的被导出的类是一个我自己写的,内容相当简单:

class CExportClass
{
public:
  int GetData() const
  {
    return m_Data;
  }

  void SetData(int Data)
  {
    m_Data = Data;
  }
protected:
private:
  int m_Data;
};

就为了导出这么一个测试类,我写了如下一个函数:

void ExportToLua()
{
  using namespace LuaPlus;
  LuaClass<CExportClass>(GLuaState)
    .create("CExportClass")
    .def("Get",&CExportClass::GetData)
    .def("Set",&CExportClass::SetData);
}

谁知一编译就报很多如下所示的错误:
LuaPlusCD.h(971): error C2780: “intLPCD::Call(RT (__cdecl *)(void),lua_State *,int)” : 应输入 3 个参数,却提供了 4个LuaPlusCD.h(702) : 参见“LPCD::Call”的声明
        为此我查了很久都没能找到问题的根源,只得将沐枫的原例Logger类拿来试一试,与我自己写的类比一比。这一试,我发现在原例中,所有的函数均没有使用const,想必肯定是这个原因,我曾经在自己的项目中也遇到过这样的问题。将const一去掉,OK,编译时以上的错误就永远地消失了!但是问题到此还没有结束,还有一个错误发生在LuaPlusHelper.h头文件的LuaConstructor::ConstructorHelper中,由于我下载的那个LuaPlusHelper.h头文件是用于LuaPlus5.0版本的,而我现在用的是LuaPlus5.1,PushStack函数已经不是LuaObject的成员了,原来的PushStack函数在LuaPlus5.1版本中已经被更名为Push了,于是乎将之改掉.现在编译连接就没有问题了!
       编译连接都通过后,我便利用导出的Logger类的来测试一下,还是按照原例中的代码来测试,一运行,没有想到出现问题了:我新建的一个Logger对象,在使用其成员函数时的this指针却不是生成时的this指针,也就是说脚本中找到的我需要的对象的地址有问题!我将此测试放到LuaPlus5.0版本中,测试也会遇到同样的问题.经过跟踪调试,发现原来是LPCD::GetObjectUserData函数有问题,原代码:
if (type == LUA_TUSERDATA)
   return lua_unboxpointer(L, 1);
其中lua_unboxpointer是一个宏,其定义为#define lua_unboxpointer(L,i)   (*(void **)(lua_touserdata(L, i)))
跟踪调试时发现lua_touserdata函数本身返回的就是我们需要的对象的正确地址,但是经过(*(void **)这么一转换就变成了另一个地址了.将之改为:
if (type == LUA_TUSERDATA)
   return lua_touserdata(L, 1);
现在编译运行,OK!全部正确!至此,整个C++成员函数的导出成功!
 

发表于 @ 2007年05月18日 09:29:00|评论(1)|编辑

新一篇: C++类模板的成员函数模板写法 | 旧一篇: VSLua工具