【Cocos2d-x lua篇002】Demo讲解之Lua和C++牵手

1. 引入头文件

我们来看看要在C++中使用Lua,需要些什么东西

  1. /*  
  2.    文件名:    HelloLua.h  
  3.    描 述:    Lua Demo 
  4.    创建人:    笨木头 (CSDN博客:http://blog.csdn.net/musicvs)  
  5.  
  6.    创建日期:   2012.12.24  
  7. */    
  8.   
  9. #ifndef __HELLO_LUA_H_  
  10. #define __HELLO_LUA_H_  
  11.   
  12. #include "cocos2d.h"  
  13.   
  14. extern "C" {  
  15. #include <lua.h>  
  16. #include <lualib.h>  
  17. #include <lauxlib.h>  
  18. };  
  19.   
  20. using namespace cocos2d;  
  21.   
  22. class HelloLua : public CCLayer {  
  23. public:  
  24.     CREATE_FUNC(HelloLua);  
  25.     virtual bool init();  
  26.   
  27.     static CCScene* scene();  
  28. };  
  29.   
  30. #endif  

看到红色粗体的代码了吗?

在这:

extern "C" {

#include <lua.h>

#include <lualib.h>

#include <lauxlib.h>

};


 

记住了,LuaC语言库,所以在C++中使用必须用extern “C”声明,让编译器知道。

有了这些,我们就能开始使用Lua了。


 

啊,对了,还少一样东西,不过这个不需要我们做了,那就是引入Lua的库,没有库,我们怎么包含头文件都没用。

不过没关系,Cocos2d-x本来就支持Lua,所以这一步我们省下了,为了保险起见,我在新建Demo项目的时候勾选了支持Lua

建议大家首先能创建一个支持LuaCocos2d-x项目,并且能编译运行,然后再继续往下看~


 

我教?我不懂~

 

2. 开始使用

来看看我们的cpp文件,我们要开始使用Lua~

  1. #include "HelloLua.h"  
  2.   
  3. CCScene* HelloLua::scene() {  
  4.     CCScene* scene = CCScene::create();  
  5.     CCLayer* layer = HelloLua::create();  
  6.     scene->addChild(layer);  
  7.   
  8.     return scene;  
  9. }  
  10.   
  11. bool HelloLua::init() {  
  12.     lua_State* pL = lua_open();  
  13.     luaopen_base(pL);  
  14.     luaopen_math(pL);  
  15.     luaopen_string(pL);  
  16.   
  17.     /* 1.执行Lua脚本,返回0代表成功 */  
  18.     /* 2.重置栈顶索引 */  
  19.     /* 3.判断栈顶的值的类型是否为String, 返回非0值代表成功 */  
  20.   /* 4.获取栈顶的值 */  
  21.     
  22.     lua_close(pL);  
  23.     return true;  
  24. }  


 

为了不一下子就一大堆代码吓坏大家,我把部分代码先删了,我们来看看现在这个代码的情况:

1) HelloLua是一个场景

2) HelloLua有一个init函数

3) 我就喜欢旁白吐槽~

4) 要使用Lua,首先要有一个lua_State,这是什么呢?我引用《游戏人工智能编程案例精粹》一书的一句话(191页):“每一个运行的脚本文件都在一个动态分配的叫做lua_State的数据结构中运行”。不明白的话,也没有关系,我们就把lua_State当成是一个Lua的身体,Lua在做任何事情的时候都不能没有身体。

5) 接下来看到几句话:luaopen_base(pL);luaopen_math(pL);luaopen_string(pL);

Lua有一些标准库,要使用这些库,就要用luaopen_**去加载这些库

6) 然后最后还有一句话:lua_close(pL),一看就知道了,用来释放内存的。

7) 旁白呢?

 

3. 执行Lua脚本

现在我们来一步步完善我们的代码,执行Lua脚本很简单,看看:

  1. bool HelloLua::init() {  
  2.     lua_State* pL = lua_open();  
  3.     luaopen_base(pL);  
  4.     luaopen_math(pL);  
  5.     luaopen_string(pL);  
  6.   
  7.     /* 1.执行Lua脚本,返回0代表成功 */  
  8.     int err = luaL_dofile(pL, "helloLua.lua");  
  9.     CCLOG("open : %d", err);  
  10.   
  11.     /* 2.重置栈顶索引 */  
  12.     lua_settop(pL, 0);  
  13.     lua_getglobal(pL, "myName");  
  14.   
  15.     /* 3.判断栈顶的值的类型是否为String, 返回非0值代表成功 */  
  16.   /* 4.获取栈顶的值 */  
  17.     
  18.     lua_close(pL);  
  19.     return true;  
  20. }  


 

我们还要新建一个lua文件,很简单,新建一个文本文件,把后缀名改为lua就行了。现在我们来创建一个helloLua.lua文件:

[plain]  view plain copy print ?
  1. -- helloLua.lua文件  
  2. myName = "beauty girl"  


 


好,lua文件也有了,在C++中只要调用luaL_dofile就能执行lua脚本了,注意了,必须把lua_State也作为参数传给luaL_dofile,前面已经说了,身体不能少。

 

 

4. 重置栈顶索引, 将全局变量放到堆栈中

大家没有发现吗?我把第2步也放出来了~

 

lua_settop(pL, 0);是为了确认让栈顶的索引置为0,因为我们操作栈的时候是根据索引来操作的。置0之后,我们入栈的第一个元素的索引就是1

那,lua_getglobal(pL, “myName”);又是什么呢?咋一看好像是从lua中取得myName这个全局变量的值,但并不是这样的,虽然最终也是这样。


 

我们之前说过了,LuaC++是不能直接通信的,要通过堆栈来通信。

因此,lua_getglobal(pL, “myName”);只是把myName放到了栈中,然后lua就会通过myName去全局表寻找,找到myName对应的字符串“beauty girl”,再放到栈中。(第01章的时候介绍过的步骤,还记得吗?不记得的建议大家去看看~


1.C++myName放到堆栈

2.lua从堆栈取得myName

3.luamyNamelua全局表查找获取myName对应的字符串,得到“beauty girl”字符串,然后再放回堆栈

4.最后C++就可以从堆栈中取得“beauty girl”字符串? 

 

5. 最后一步,C++取得字符串

我们来看看完整的代码:

  1. bool HelloLua::init() {  
  2.     lua_State* pL = lua_open();  
  3.     luaopen_base(pL);  
  4.     luaopen_math(pL);  
  5.     luaopen_string(pL);  
  6.   
  7.     /* 1.执行Lua脚本,返回0代表成功 */  
  8.     int err = luaL_dofile(pL, "helloLua.lua");  
  9.     CCLOG("open : %d", err);  
  10.   
  11.     /* 2.重置栈顶索引 */  
  12.     lua_settop(pL, 0);  
  13.     lua_getglobal(pL, "myName");  
  14.   
  15.     /* 3.判断栈顶的值的类型是否为String, 返回非0值代表成功 */  
  16.     int isstr = lua_isstring(pL, 1);  
  17.     CCLOG("isstr = %d", isstr);  
  18.   
  19.     /* 4.获取栈顶的值 */  
  20.     const char* str = lua_tostring(pL, 1);  
  21.     CCLOG("getStr = %s", str);  
  22.   
  23.     lua_close(pL);  
  24.     return true;  
  25. }  


 

lua_getglobal已经完成了很多工作了,现在堆栈上就放着“beauty girl”字符串,我们只要去取就可以了。

获取堆栈的值有很多种方法,分别对应不同的变量类型:

lua_toboolean

lua_toNumber

lua_tocfunction

lua_tostring

我就不全部举例了,现在我们要用lua_tostring来获取栈顶的值。

最后,在AppDelegate.cpp中把默认启动场景设为我们的HelloLua场景,用调试模式运行项目,将看到以下日志:

open : 0

isstr = 1

getStr = beauty girl

好,本章到此结...

 

对了对了,Lua还提供了很多函数供我们判断堆栈中的变量类型,比如lua_isstringlua_isnumber等等,和lua_tostring等函数是对应的。返回非0值表示类型正确。

一般在取值之前都要判断一下,不能程序很可能意外崩溃~

 

~本章到此结束~


那就是用lua_pop(pL, 1); 可以清除指定堆栈上的数据~


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值