对“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++中unordered_set哈希集合的实现

《C++中unordered_set哈希集合的实现》std::unordered_set是C++标准库中的无序关联容器,基于哈希表实现,具有元素唯一性和无序性特点,本文就来详细的介绍一下unorder... 目录一、概述二、头文件与命名空间三、常用方法与示例1. 构造与析构2. 迭代器与遍历3. 容量相关4

C++中悬垂引用(Dangling Reference) 的实现

《C++中悬垂引用(DanglingReference)的实现》C++中的悬垂引用指引用绑定的对象被销毁后引用仍存在的情况,会导致访问无效内存,下面就来详细的介绍一下产生的原因以及如何避免,感兴趣... 目录悬垂引用的产生原因1. 引用绑定到局部变量,变量超出作用域后销毁2. 引用绑定到动态分配的对象,对象

Nginx分布式部署流程分析

《Nginx分布式部署流程分析》文章介绍Nginx在分布式部署中的反向代理和负载均衡作用,用于分发请求、减轻服务器压力及解决session共享问题,涵盖配置方法、策略及Java项目应用,并提及分布式事... 目录分布式部署NginxJava中的代理代理分为正向代理和反向代理正向代理反向代理Nginx应用场景

IDEA和GIT关于文件中LF和CRLF问题及解决

《IDEA和GIT关于文件中LF和CRLF问题及解决》文章总结:因IDEA默认使用CRLF换行符导致Shell脚本在Linux运行报错,需在编辑器和Git中统一为LF,通过调整Git的core.aut... 目录问题描述问题思考解决过程总结问题描述项目软件安装shell脚本上git仓库管理,但拉取后,上l

Redis中的有序集合zset从使用到原理分析

《Redis中的有序集合zset从使用到原理分析》Redis有序集合(zset)是字符串与分值的有序映射,通过跳跃表和哈希表结合实现高效有序性管理,适用于排行榜、延迟队列等场景,其时间复杂度低,内存占... 目录开篇:排行榜背后的秘密一、zset的基本使用1.1 常用命令1.2 Java客户端示例二、zse

Redis中的AOF原理及分析

《Redis中的AOF原理及分析》Redis的AOF通过记录所有写操作命令实现持久化,支持always/everysec/no三种同步策略,重写机制优化文件体积,与RDB结合可平衡数据安全与恢复效率... 目录开篇:从日记本到AOF一、AOF的基本执行流程1. 命令执行与记录2. AOF重写机制二、AOF的

idea npm install很慢问题及解决(nodejs)

《ideanpminstall很慢问题及解决(nodejs)》npm安装速度慢可通过配置国内镜像源(如淘宝)、清理缓存及切换工具解决,建议设置全局镜像(npmconfigsetregistryht... 目录idea npm install很慢(nodejs)配置国内镜像源清理缓存总结idea npm in

pycharm跑python项目易出错的问题总结

《pycharm跑python项目易出错的问题总结》:本文主要介绍pycharm跑python项目易出错问题的相关资料,当你在PyCharm中运行Python程序时遇到报错,可以按照以下步骤进行排... 1. 一定不要在pycharm终端里面创建环境安装别人的项目子模块等,有可能出现的问题就是你不报错都安装

idea突然报错Malformed \uxxxx encoding问题及解决

《idea突然报错Malformeduxxxxencoding问题及解决》Maven项目在切换Git分支时报错,提示project元素为描述符根元素,解决方法:删除Maven仓库中的resolv... 目www.chinasem.cn录问题解决方式总结问题idea 上的 maven China编程项目突然报错,是

使用EasyPoi快速导出Word文档功能的实现步骤

《使用EasyPoi快速导出Word文档功能的实现步骤》EasyPoi是一个基于ApachePOI的开源Java工具库,旨在简化Excel和Word文档的操作,本文将详细介绍如何使用EasyPoi快速... 目录一、准备工作1、引入依赖二、准备好一个word模版文件三、编写导出方法的工具类四、在Export