tolua++初探六

    这是学习tolua++的最后一篇了。在这一篇里完成一个稍微复杂一点的例子(^_^其实还是很简单)。     导出三个类CBase、CDerived1、CDerived2到lua,导出两个函数toDerived1、toDerived2。lua脚本中声明两个函数Derived1Test和Derived2Test,我们在C++中调用。Derived1Test和Derived2Test会调用toDerived*对其参数进行向下转换(从CBase转到CDerived*),然后调用派生类的方法做一些测试。     基本上和前面几个例子类似,新增加的部分是在C++中调用Lua脚本里定义的函数。这牵涉到虚拟栈的操作,后面会解释一下。     还是老样子,先把实际的头文件列出来。tlclass.h如下:

#ifndef __TOLUACLASS_h

#define  __TOLUACLASS_h

class  CBase

{

public:

    CBase()
{}

    
virtual ~CBase()

    

    
void ShowMessage(){printf("BaseClass );}

    
staticchar* ClassName()returnCBase; }

    

}
;

 CDerived1 :  public  CBase

:

    CDerived1()

    
CDerived1()

    

    
 ShowMessage()Derived1Class

    
 ShowDerived1()show derived11111

}
 CDerived2 :   CBase

:

    CDerived2()
{m_nNumber=0;}

    
CDerived2()

    
 ShowMessage()Derived2Class

    
 ShowDerived2()show derived22222

    
 SetNumber(int num){ m_nNumber  num; }

    
  GetNumber()  m_nNumber; }

protected:

    
 m_nNumber;

}
extern  CDerived1  *  toDerived1( void    p);

 CDerived2   toDerived2( #endif
        tlclass.pkg如下:
$#include  " tlclass.h

 CBase

:

    
 ShowMessage();

    
 ClassName();

    

}
 CBase

:

    

    
 ShowDerived1();

}
 CBase

 ShowDerived2();

    
 num);

    
  GetNumber();

    

}
;

CDerived1 
 p);

CDerived2 
 p);
    这次多定义了两个函数toDerived1和toDerived2,全局的。我们也可以把他们直接放在类中,或者一个MODULE中。module大概是类似的namespace的东西,把一堆杂七杂八的家什如变量、常量、函数、类实例等放在一起,在lua中通过"."来访问。下面是手册中的例子:
module mod

{

 
#define N

 
extern var;

 
 func (...):

}
 
    这样我们可以在lua中用mod.N,mod.var,mod.func来访问其成员。     原本toDerived*的参数是CBase*,但是从C++向Lua函数传参数的时候我调用了lua_pushlightuserdata,结果在脚本中报错,说toDerived*应当接受CBase*而非userdata。于是干脆把参数修改成void*,这下lua不再叫唤了。     好了,到了列出驱动文件的时候了。CallLuaFunc.cpp:
#include  lua.hpp

#include 

CDerived1 
 p)

{

    
 dynamic_cast<CDerived1*>((CBase)p);

}

CDerived2 
 p)

CDerived2int  tolua_calllua_open(lua_State );

 _tmain(  argc, _TCHAR  argv[])

{

    lua_State 
 L  luaL_newstate();

    luaopen_base(L);

    tolua_calllua_open(L);

    luaL_dofile(L, 
../scripts/CallLuaFunc.lua);

    
//call lua function

    CBase  p1 new CDerived1();

    CBase 
 p2  CDerived2();

    
call Derived1Test    lua_getglobal(L,0);">Derived1Test);

    lua_pushlightuserdata(L, p1);

    
if( lua_pcall(L,0);">1,0);">) != )

    
{

        fprintf(stderr,0);">call Derived1Test failed:%s
-));

    }

    
call Derived2TestDerived2Test

    
 )

    
call Derived2Test failed:%s

    printf(
This info is print in C++! CDerived2.GetNumber()=%d )p2)->GetNumber());

    delete p1;

    delete p2;

    lua_close(L);

    
;

}

    这次驱动文件有了点新的变化:1)两个全局导出函数;2)调用lua函数的代码。分开来看。     导出函数toDerived*很简单,只是调用dynamic_cast来向下转换而已。如果转换失败,dynamic_cast会返回null。当我们要从基类指针转换到派生类指针时,最好用dynamci_cast,直接强制转换是危险的,除非你明确的知道某个指针指向的对象是什么。     在C++中调用lua脚本的函数大概分为三步:     a..找到函数并入栈;(这里是 lua_getglobal(L,0);">);)     b..参数入栈;(这里是 lua_pushlightuserdata(L, p1);)     c..调用lua_pcall进行实际调用     第一步不必说了;第二步可以传递任意个任意类型的参数,lua_pushnumber,lua_pushstring,lua_pushboolean等等可以调用;第三步是调用lua_pcall,lua_pcall第一个参数是lua_State*,这是我们的工作环境了。第二参数是要传递的参数个数,我们这里是1;第三个参数是lua函数返回的结果个数,我们的lua函数不返回结果,设为0。第四个参数是比较复杂,为0时指lua_pcall会在调用失败时把原始错误信息放到栈上;其它值代表栈的索引,该索引处放了一个错误处理函数,lua_pcall失败时会根据这个索引去调用该函数。     调用失败的时候我只是简单地打印一条出错信息,这个错误码放在栈顶,我们用lua_tostring(L,-1)访问并转换为字符串。可以修改下驱动代码,比如把第二次调用传入p1,这样就可以看见错误信息。     最后我还是在C++代码中打印了下CDerived2对象的值,以验证lua和C++中访问的是同一个对象。    Lua和C++的交互都是通过栈,所以要写交互部分的代码就要不停的出栈入栈,烦死个人。不过这也是Lua灵活的地方。牛人啊,顶礼膜拜吧。     有时间要好好研究下《Programming Lua》,CSDN上有中文版的,lua官方网站上有英文的。虽然这个是针对5.0版本的lua,但绝大部分东西还是有用的。针对5.1.3的第二版已经出了,可惜我在网上没有找到链接,哪位看到分享一下。     下面看看lua文件callluafunc.lua吧:

print( " now in CallLuaFunc.lua! )

-- lua function to test CDerived1, CDerived2, they ' ll be called from C++

function Derived1Test(e) 

    d1 
=  toDerived1(e);

    
if  d1 then 

        d1:ShowMessage(); 

        d1:ShowDerived1(); 

    
else

        print(
invalid d1(nil)! );

    end

end

function Derived2Test(e) 

        d2 
 toDerived2(e); 

    
 d2 then

        d2:ShowMessage(); 

        d2:ShowDerived2(); 

        d2:SetNumber(
180 );

        print(d2:GetNumber());

    
invalid d2(nil) );

    end

end
    lua中定义了函数Derived1Test和Derived2Test。上面的版本已经不会导致出错信息了,原始的Derived*Test函数如下:
 toDerived1(e);

    d1:ShowMessage(); 

    d1:ShowDerived1(); 

end

function Derived2Test(e) 

    d2   toDerived2(e); 

    d2:ShowMessage(); 

    d2:ShowDerived2(); 

    d2:SetNumber(
);

    print(d2:GetNumber());

end
    原始的Derived*Test函数没有错误检查,所以从C++中用lua_pcall调用时可能会产生错误信息。     又想了下,这样的类型转换太复杂了,tolua++提供了转换机制,可以用的。CEGUI用的就是。     tolua++生成了一些工具函数,在tolua为名的module中。其中tolua.cast就是用来做类型转换的。只需要改动tlclass.pkg文件,加入下面的代码:

$[

testHelper
= {}

function testHelper.toDerived1(e)

    
return  tolua.cast(e,  CDerived1 )

end

function testHelper.toDerived2(e)

    
CDerived2 )

end

$]

    $[和$]结合,用来直接插入lua代码。我生成一个名为testHelper的table,给testHelper添加两个转换函数。     重新用tolua++编译,再编译工程。就可以在脚本中调用testHelper.toDerived*来转换了。     下面是更改后的 callluafunc.lua文件:
)

function Derived1Test(e) 

    d1 
 testHelper.toDerived1(e);

    
 d1 then 

        d1:ShowMessage(); 

        d1:ShowDerived1(); 

    

        print(
);

    end

end

function Derived2Test(e) 

        d2 
 testHelper.toDerived2(e); 

    
 d2 then

        d2:ShowMessage(); 

        d2:ShowDerived2(); 

        d2:SetNumber(
);

        print(d2:GetNumber());

    
);

    end

end
    仅仅是将toDerived*调用转换成了testHelper.toDerived*。运行了一下,结果是正常的。          好啦好啦,就到这里啦。     通过两天的学习,我已经确定可以在项目中使用tolua++了,它是"AS IS"的,可以用于任何目的。到目前位置所演示的一些特性,可以满足我的需要。     嗯,有些未完成的东西,比如UNICODE、多线程环境下对lua的调用等,慢慢用到了再说吧。    

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐


1.github代码实践源代码是lua脚本语言,下载th之后运行thmain.lua-netTypevgg-savevgg_cifar10/-S0.0001,报错: 试看看安装lua:报错了,参考这篇文章:ubuntu18.04安装lua的步骤以及出现的问题_weixin_41355132的博客-CSDN博客问题解决,安装成功:情况并没有好转,出现相
此文为搬运帖,原帖地址https://www.cnblogs.com/zwywilliam/p/5999924.html前言在看了uwa之前发布的《Unity项目常见Lua解决方案性能比较》,决定动手写一篇关于lua+unity方案的性能优化文。整合lua是目前最强大的unity热更新方案,毕竟这是唯一可以支持ios热更新的办法。然而作
Rime输入法通过定义lua文件,可以实现获取当前时间日期的功能。1.TIMERime是一款可以高度自定义的输入法,相关教程可以查看往期文章,关于时间获取是指输入一个指定关键字,输出当前时间,效果如下(我定义了time关键字):实现如下:①在用户文件夹中新建一个rime.lua文件加入如下代码 ti
localfunctiongenerate_action(params)localscale_action=cc.ScaleTo:create(params.time,params.scale_x,params.scale_y)localfade_action=cc.FadeIn:create(params.time)returncc.Spawn:create(scale_action,fade_action)end
2022年1月11日13:57:45 官方:https://opm.openresty.org/官方文档:https://opm.openresty.org/docs#table-of-contents为什么建议使用opm不建议使用luarocks?http://openresty.org/cn/using-luarocks.html官方解释:请注意!LuaRocks并不是OpenResty官方推荐的装包方式。LuaRoc
在Lua中的table(表),就像c#中的HashMap(哈希表),key和value一一对应。元表:table的一个操作的拓展,里面包含关联了对应的方法,元方法就是其中一个。元方法:当你通过键来访问table的时候,如果这个键没有值,那么Lua就会寻找该table的metatable(假定有metatable)中的__index键。如果__inde
表排序:table.sort(list[,comp])参数list:指定表,可选参数comp:排序函数,无参数时通常按升序排序。排序函数针对表中连续的序列,其间不可以存在空洞或nil,排序函数需要两个形参(对应表中每次参加比较的两个数据),需要一个比较两个形参表达式的返回值,不能含有等于关系,例如>=,<=,==。do
一、安装lua环境1.1安装依赖包[root@centos7~]#yuminstallgccreadline-devel1.2下线lua源码包并解压[root@centos7~]#wgethttp://www.lua.org/ftp/lua-5.3.5.tar.gz[root@centos7~]#tarxvflua-5.3.5.tar.gz-C/usr/local/src1.3进行编译[root@centos7~]
官网OpenResty® 是一个基于 Nginx 与Lua的高性能Web平台,其内部集成了大量精良的Lua库、第三方模块以及大多数的依赖项。用于方便地搭建能够处理超高并发、扩展性极高的动态Web应用、Web服务和动态网关。OpenResty® 通过汇聚各种设计精良的 Nginx 模块(主要由
表参考《lua程序设计》可以认为,表是一种动态分配的对象,程序只能操作指向表的引用(或指针)。除此以外,Lua语言不会进行隐藏的拷贝(hiddencopies)或创建新的表--创建表a={}--创建空表k="x"a[k]=10--键“x”值10a[20]="great"--键20值“great”print(a["x"])-->10
https://github.com/galenho/crossover.git一个跨平台的lua游戏服务器开发框架,该框架采用多线程并发来处理消息,开发者只需要调用相应的接口函数并绑定相应的回调函数即可,在逻辑层表现为单线程的开发模式,使开发者易用,易调试,易维护,易扩展,同时拥有快速的响应能力。   框架使用面
参考链接:https://www.runoob.com/lua/lua-metatables.htmlhttps://www.jianshu.com/p/cb945e7073a3 元表是一个table,可以让我们改变table的行为,每个行为有对应的元方法例如,对table进行设置键值,查找键值,运算等,就会触发对应的元方法1--__index:table被访问时,如果找不到这
https://github.com/yuin/gopher-luahttps://github.com/yuin/gopher-lua Lua5.1ReferenceManual-contentshttp://www.lua.org/manual/5.1/ go中使用luapackagemainimport( lua"github.com/yuin/gopher-lua")funcmain(){ l:=lua.NewState() d
编译问题不要留到运行时才跑出来啊。早上9:00-中午3:00,6个小时,服了自己了。 写了一个测试,springboot+redis+lua执行到redisTemplate.execute(redisScript,idList)的时候一直报错,integer无法转换为string。我一直以为是lua脚本写错了,翻文档翻过来又翻过去,写法变了又变,还是解
        。。是字符串连接符,字典用=号连接,  注意fordoend都是连一起,  注意ifthen,  如果local在函数里,是可以访问,非local出了函数一样能用,  doend代码块也是一样,    注意点号表示,只能key是字符串,  注意括号不是必须
C语言与Lua之间的相互调用详解写一个C调用Lua的Demo编译运行C语言调用Lua编译问题总结正确的编译命令问题1:缺少-lm参数问题2:缺少-ldl参数​1、为什么会出现undefinedreferenceto‘xxxxx’错误?​2、-l参数和-L参数写一个C调用Lua的Demo编译运行add.c内容//你需要
1、动态输出打开E:\study\openresty\openresty-1.19.9.1-win64目录下的confginx.conf文件在server中增加一下代码 location/hello{ default_typetext/html; content_by_lua'ngx.say("<p>hello,world</p>")'; }运行后,效果如下图localhost
参见:lipp/lua-websockets:WebsocketsforLua.(github.com)github网址可能需手动转换lipp.github.com/lua-websockets/>github.com/lipp/lua-websocketswebsockets为底层的类似于TCP、UDP的socket(实现上基于更底层的socket),不同于上层的webserver服务端(Service)需并行地支持多
lua发送消息到rabbitmq,我们选择类库lua-resty-rabbitmqstomp 来完成这个任务。类库安装:进入nginx.conf中 lua_package_path 中对应的目录下的resty目录(没有则创建),执行:wget-chttps:/aw.githubusercontent.com/wingify/lua-resty-rabbitmqstomp/master/libes
1Lua介绍Lua是一门以其性能著称的脚本语言,被广泛应用在很多方面。Lua一般用于嵌入式应用,现在越来越多应用于游戏当中,魔兽世界,愤怒的小鸟都有用到。优势Lua极易嵌入到其他程序,可当做一种配置语言。提升应用性能,比如:游戏脚本,nginx,wireshark的脚本兼容性强,可以直接使用C