现在还是个新手,对很多还不是很了解,所以语言估计会有点晦涩,望原谅啊!!
<pre name="code" class="html">TOLUA_API int tolua_isusertable (lua_State* L,int lo,const char* type,int def,tolua_Error* err) { if (def && lua_gettop(L)<abs(lo)) return 1; if (lua_isusertable(L,lo,type)) return 1; err->index = lo; err->array = 0; err->type = type; return 0; }
static int lua_isusertable (lua_State* L,const char* type) { int r = 0; if (lo < 0) lo = lua_gettop(L)+lo+1; lua_pushvalue(L,lo); lua_rawget(L,LUA_REGISTRYINDEX); /* get registry[t] */ if (lua_isstring(L,-1)) { r = strcmp(lua_tostring(L,-1),type)==0; if (!r) { /* try const */ lua_pushstring(L,"const "); lua_insert(L,-2); lua_concat(L,2); r = lua_isstring(L,-1) && strcmp(lua_tostring(L,type)==0; } } lua_pop(L,1); return r; }
代码的意思很简单,无非就是判断栈中 loc位置的数据是不是usertable。代码大概的思路就是从栈的loc位置取出值A,来在注册表中找到值B,然后对B进行字符串比较判断是不是type类型或者const type类型。所以很明显A就是type或者const type对应的table。那什么时候会调用tolua_isusertable呢?
索性就搜索一下tolua_isusertable 到底被哪里引用到了。看到基本上都是create方法中引用了,还有一些其他的方法,仔细观察这些函数,发现都是静态函数,所以就猜测 是不是静态函数在生成 lua 和cocos中间层代码的时候会被生成?我带着疑问就在自定义类中也定义了个静态方法,发现果然生成了tolua_isusertable。
静态方法可以通过类来调用,所以在执行之前要判断lua传递过来的是不是该类的类型。这就是tolua_isusertable的作用。而其它的成员函数是判断传递过来的是不是userdata,并且该userdata是不是该类的类型 。
举个例子吧:
<span style="font-family:KaiTi_GB2312;font-size:14px;">local test = ccc.MyClass:create()</span>
这里ccc.MyClass就是个table,是传递到cocos的loc位置的数据,create在C++的MyClass 中是静态函数。
总结:tolua_isusertable用在lua直接通过表(C++的类)来调用的函数中,判断这个表是不是对应的C++类型。而tolua_isuserdata用在通过对象调用的函数中。
那么问题来了,我可以通过lua变量来访问静态函数么?我试了下,发现不行也是坑爹货。
<pre name="code" class="html">local test = ccc.MyClass:create() print("the first cocos bind lua result "..test:foo(2)) print(test:getData())
程序崩溃。。。
那就没有办法来解决这个问题吗?其实要改的话,也挺简单。先贴下getData的中间代码吧:
<pre name="code" class="html">int lua_myclass_MyClass_getData(lua_State* tolua_S) { int argc = 0; bool ok = true; #if COCOS2D_DEBUG >= 1 tolua_Error tolua_err; #endif #if COCOS2D_DEBUG >= 1 if (!tolua_isusertable(tolua_S,1,"gls.MyClass",&tolua_err))) goto tolua_lerror; #endif argc = lua_gettop(tolua_S) - 1; if (argc == 0) { if(!ok) return 0; int ret = GameLogic::MyClass::getData(); tolua_pushnumber(tolua_S,(lua_Number)ret); return 1; } CCLOG("%s has wrong number of arguments: %d,was expecting %d\n ","getData",argc,0); return 0; #if COCOS2D_DEBUG >= 1 tolua_lerror: tolua_error(tolua_S,"#ferror in function 'lua_myclass_MyClass_getData'.",&tolua_err); #endif return 0; }
if (!tolua_isusertable(tolua_S,&tolua_err))) goto tolua_lerror;
程序就是在这里崩溃的,tolua发现传递过来的不是个table,其实它是个userdata,所以报错啦!!我们改的也是这里,加个判断是否是userdata就可以了嘛!!
将那行改成:
if (!tolua_isusertable(tolua_S,&tolua_err) && !tolua_isusertype(tolua_S,&tolua_err)) goto tolua_lerror;