对“tolua++导出C++子类后在Lua中无法添加新成员”这一问题的初步分析【转】

2023-11-23 06:58

本文主要是介绍对“tolua++导出C++子类后在Lua中无法添加新成员”这一问题的初步分析【转】,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

原文地址:http://my.oschina.net/SunLightJuly/blog/262880

一、问题的起因

    问题首先是在Quick-x中发现的,原因是希望为CCNode添加一个Lua自定义方法就会出错:

1function CCNode:myMethod()
2end

    运行错误提示是 [LUA ERROR] stack overflow,在Player上运行时甚至直接抛出异常提示框必须直接关闭程序。

    进一步调试,有以下结果:

    1.此问题只出现在导出的子类中,如CCObject就不会有问题,而包括CCNode在内所有继承CCObject的类就有问题

    2.并不仅仅是新加方法有问题,新加成员变量也是有问题的

    因此,这一问题可以描述为:tolua++导出的C++子类,在Lua中无法添加新成员。


二、问题的分析

    由于是无法添加新成员,首先考虑的是__newindex元方法是否有问题。

    分析tolua++的代码,可以看到在tolua_classevents函数中,将__newindex设置为了class_newindex_event函数:

01static int class_newindex_event (lua_State* L)
02{
03    int t = lua_type(L,1);
04 
05    if (t == LUA_TUSERDATA)
06    {
07        //此处代码略去
08    }
09    else if (t== LUA_TTABLE)
10    {
11        module_newindex_event(L);
12    }
13    return 0;
14}

    当CCObject和它的子类添加新成员时,这里的t值是LUA_TTABLE,因此会调用module_newindex_event函数:

01static int module_newindex_event (lua_State* L)
02{
03    lua_pushstring(L,".set");
04    lua_rawget(L,-4);
05    if (lua_istable(L,-1))
06    {
07        //此处代码略去
08    }
09    /* call old newindex meta event */
10    if (lua_getmetatable(L,1) && lua_getmetatable(L,-1))
11    {
12        lua_pushstring(L,"__newindex");
13        lua_rawget(L,-2);
14        if (lua_isfunction(L,-1))
15        {
16            lua_pushvalue(L,1);
17            lua_pushvalue(L,2);
18            lua_pushvalue(L,3);
19            lua_call(L,3,0);
20        }
21    }
22    lua_settop(L,3);
23    lua_rawset(L,-3);
24    return 0;
25}

    可以看出这里是取metatable里的.set表,并对新的成员进行赋值。

   第一个判断,CCObject和它的子类都不会进入,没有影响。

   但第二个判断情况就不同了,CCObject不会进入,它的子类是会进入的。从代码写法上看,这里似乎是希望调用其父类的__newindex,但这样是很有问题的。

    首先现在这样的写法,取到的仍然是子类自己的__newindex,因此会再调用一次class_newindex_event函数,而class_newindex_event自然又会调用module_newindex_event,结果又回到了这段代码当中!这样,函数不断递归调用,永无休止,直到堆栈溢出为止,这就是出错为stack overflow的原因。

    其次我还没有想明白的是,即使是父类的__newindex,这里似乎也没有调用的必要。因为子类要是定义一个新成员是在父类里生效的话,那问题就大了。所以这里我认为直接调用lua_rawset应该就可以了。


三、临时解决方案

    基于以上分析,我认为问题就在于module_newindex_event的处理上。我尝试不调用module_newindex_event,而直接对新成员进行赋值操作,即修改class_newindex_event函数的相关代码如下:

1    else if (t== LUA_TTABLE)
2    {
3

//        module_newindex_event(L);

  //quick新版是做了如下的改动【臧旭】  

  //lua_pushstring(L,".set");
  //lua_rawget(L,-4);
  //lua_settop(L,3);
  //lua_rawset(L,-3);

4        lua_settop(L,3);
5        lua_rawset(L,-3);
6    }

   修改的这两句相当于在lua里面直接调用rawset对新成员进行赋值操作。

    修改后,在Lua中运行

1function CCNode:testFunc()
2  print("----------CCNode.testFunc------Entry now!!!")
3end
4 
5function MainScene:ctor()
6    local s = display.newSprite("Logo.png", display.cx, display.cy)
7    self:addChild(s)
8    s:testFunc()
9end

    上述代码是可以正常运行并得到预想中的输出的。

    不过目前这只是临时方案。我不知道tolua++原来这样写代码是不是另有原因。后面我将再仔细考虑清楚,这样的修改会不会有什么负作用。

(2014.5.30加注:quick-x的2.2.3版本目前已经采用了这一修改方案)

这篇关于对“tolua++导出C++子类后在Lua中无法添加新成员”这一问题的初步分析【转】的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/416162

相关文章

C++如何通过Qt反射机制实现数据类序列化

《C++如何通过Qt反射机制实现数据类序列化》在C++工程中经常需要使用数据类,并对数据类进行存储、打印、调试等操作,所以本文就来聊聊C++如何通过Qt反射机制实现数据类序列化吧... 目录设计预期设计思路代码实现使用方法在 C++ 工程中经常需要使用数据类,并对数据类进行存储、打印、调试等操作。由于数据类

usb接口驱动异常问题常用解决方案

《usb接口驱动异常问题常用解决方案》当遇到USB接口驱动异常时,可以通过多种方法来解决,其中主要就包括重装USB控制器、禁用USB选择性暂停设置、更新或安装新的主板驱动等... usb接口驱动异常怎么办,USB接口驱动异常是常见问题,通常由驱动损坏、系统更新冲突、硬件故障或电源管理设置导致。以下是常用解决

Mysql如何解决死锁问题

《Mysql如何解决死锁问题》:本文主要介绍Mysql如何解决死锁问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录【一】mysql中锁分类和加锁情况【1】按锁的粒度分类全局锁表级锁行级锁【2】按锁的模式分类【二】加锁方式的影响因素【三】Mysql的死锁情况【1

SpringBoot内嵌Tomcat临时目录问题及解决

《SpringBoot内嵌Tomcat临时目录问题及解决》:本文主要介绍SpringBoot内嵌Tomcat临时目录问题及解决,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,... 目录SprinjavascriptgBoot内嵌Tomcat临时目录问题1.背景2.方案3.代码中配置t

SpringBoot使用GZIP压缩反回数据问题

《SpringBoot使用GZIP压缩反回数据问题》:本文主要介绍SpringBoot使用GZIP压缩反回数据问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录SpringBoot使用GZIP压缩反回数据1、初识gzip2、gzip是什么,可以干什么?3、Spr

Java程序进程起来了但是不打印日志的原因分析

《Java程序进程起来了但是不打印日志的原因分析》:本文主要介绍Java程序进程起来了但是不打印日志的原因分析,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录Java程序进程起来了但是不打印日志的原因1、日志配置问题2、日志文件权限问题3、日志文件路径问题4、程序

Linux下如何使用C++获取硬件信息

《Linux下如何使用C++获取硬件信息》这篇文章主要为大家详细介绍了如何使用C++实现获取CPU,主板,磁盘,BIOS信息等硬件信息,文中的示例代码讲解详细,感兴趣的小伙伴可以了解下... 目录方法获取CPU信息:读取"/proc/cpuinfo"文件获取磁盘信息:读取"/proc/diskstats"文

Java使用ANTLR4对Lua脚本语法校验详解

《Java使用ANTLR4对Lua脚本语法校验详解》ANTLR是一个强大的解析器生成器,用于读取、处理、执行或翻译结构化文本或二进制文件,下面就跟随小编一起看看Java如何使用ANTLR4对Lua脚本... 目录什么是ANTLR?第一个例子ANTLR4 的工作流程Lua脚本语法校验准备一个Lua Gramm

Java字符串操作技巧之语法、示例与应用场景分析

《Java字符串操作技巧之语法、示例与应用场景分析》在Java算法题和日常开发中,字符串处理是必备的核心技能,本文全面梳理Java中字符串的常用操作语法,结合代码示例、应用场景和避坑指南,可快速掌握字... 目录引言1. 基础操作1.1 创建字符串1.2 获取长度1.3 访问字符2. 字符串处理2.1 子字

C++使用printf语句实现进制转换的示例代码

《C++使用printf语句实现进制转换的示例代码》在C语言中,printf函数可以直接实现部分进制转换功能,通过格式说明符(formatspecifier)快速输出不同进制的数值,下面给大家分享C+... 目录一、printf 原生支持的进制转换1. 十进制、八进制、十六进制转换2. 显示进制前缀3. 指