Lua使用心得(1)

 

这几天研究了一下lua,主要关注的是lua和vc之间的整合,把代码都写好放在VC宿主程序里,然后在lua里调用宿主程序的这些代码(或者叫接口、组件,随便你怎么叫),希望能用脚本来控制主程序的行为。这实际上也是一种把业务分离,用脚本控制的架构,可能有些人把这种脚本叫做业务引擎,工作流等。

 

为什么选择lua?

因为它是一个能和C/C++结合得很紧的脚本语言,而我们的程序是用VC++ 写的;另外一点是因为它的名气,连WOW都用lua来提供API让玩家修改其游戏行为,那我是找不到什么理由拒绝它了。

 

Lua是什么?在哪里获取LUA?

详细的不说了,在网上一搜大把,只说一下它的官网吧:www.lua.org,在这里可以查到lua的应用,lua发布的版本,我用的是5.1.4,下载的是源代码的版本。

 

LUA和VC MFC的整合?

  • 1、 包含LUA:要使用LUA,当然要先把它包含进我们的工程里,可以有lib/dll方式、也可以用静态lib方式,当然也可以把整个lua的代码放进我们的工程,然后编译,因为lua只有几百K,很小。。。包含整个代码的方法我说一下:
  • a) 在VC MFC新建一个工程(例如Dialog base工程)
  • b) 去到工程里的文件tab页,新建一个文件夹,然后把所有lua里的.c、.h文件包含进来,注意有几个不用包含,lua.c、wmain.c、luac.c,包含进来之后,选中这个文件夹下面的所有.c文件,然后右键选setting,选择Not using precompiler file,具体自己找找哈。这步做完就马上编译一下,应该是没问题的了!
  • c) 还有动态库和静态lib两种方式把lua包含进工程里的,自己可以尝试一下。
  • 2、 在某个地方(我是在MFCLua1Dlg.cpp文件开始处)包含lua的头文件,并声明一个lua_state指针,如下:

extern "C"

{

#include "lua.h"

#include "lualib.h"

#include "lauxlib.h"

}

lua_State *lua;

 

在某个适当的地方(我是在OnInitDialog里)调用下面一段代码,这段代码的作用是打开一些必要的库:

       lua = lua_open (); 

       if(lua)

       {

              luaopen_base (lua);

              luaopen_table (lua);

              luaopen_string (lua);

              luaopen_math (lua);

              luaopen_debug (lua);

              //luaopen_io (lua);

       }

 

用完lua的时候,调用下面一句来关闭lua库:

lua_close (lua);

             

好了,到现在为止,lua已经完全变成我们程序的一部分了,试着编译一下,看看能不能顺利通过。。。

 

LUA和MFC的交互?

 

Lua变成我们程序的一部分之后,我们还要使用它,要记住我们的目标是用脚本程序控制我们宿主程序的执行流程,那我们就要完成两步,一是用mfc程序调用lua的函数,二是用lua调用mfc的函数,下面的内容对于初学者可能会开始有点难理解了,请打醒十二分精神,我会尽量简单的说。。。

 

1、 mfc调用lua的函数,这里用到一个StackDump的函数,是关于主程序和lua的交互栈的问题,下面会对交互栈的问题专门说明。

 

首先我们用记事本建立一个test.lua,内容是一个相加函数:

 

function add ( x,y )

return x + y;

end

 

然后再VC里调用它,如下的一段代码,看这段代码的时候,先把StackDump函数忽略,只需要知道它是一个输出lua和vc交互栈内容的函数,对了,你可以新建一个button的click函数,然后把这段代码放进去:

 

StackDump(lua);   

luaL_dofile(lua,"test.lua");     // 解释分析lua文件

StackDump(lua);

lua_getglobal(lua,"add");       // 取到一个全局标号add,取的同时会把add函数压栈

StackDump(lua);   

lua_pushnumber(lua,1);        // 把第一个参数压入栈里

StackDump(lua);

lua_pushnumber(lua,2);        // 第二个参数压栈

StackDump(lua);

//lua_call(lua,2,1);

if(lua_pcall(lua,1,0) != 0)        // 执行add函数

{

        AfxMessageBox("lua_pcall error!");            

return;

}

StackDump(lua);

int d = (int)lua_tonumber(lua,-1);        // 函数执行完了,执行结果被压栈,所以取得最顶端的一个数就是结果值,-1就是指取栈顶的值

CString str;

str.Format("%d",d);

AfxMessageBox(str);

StackDump(lua);

lua_pop(lua,1);      // 把值从栈里清除,pop(弹出)一个值

StackDump(lua);

 

好好分析一下这段代码,我们大概知道调用lua函数的一个过程是:dofile--〉函数名压栈--〉参数依序压栈--〉lua_pcall执行(执行结果压栈)--〉取出执行结果(如果有多个,就从栈里取出多个。。。),这样我们就能很轻松的调用到lua里的函数,其实就是要知道栈里发生了什么。。。

 

2、 lua调用MFC函数,比如我们想在lua里调用一个Msg函数,能弹出一个窗口来显示我们想显示的字串,然后返回值是1个"MsgOK!"字串。

 

lua文件是这样的,第一句是调用Msg函数,第二句是测试返回的字串是不是"MsgOK!":

c = Msg ("123");

Msg(c);

 

MFC程序里是这样的:

首先写一个将要被导出的函数,很多文章叫它为粘合函数(glue function):

static int Msg(lua_State* L)

{

const char *s1 = luaL_checkstring(L,1);     // 测试第一个参数是否为字串形式,并取得这个字串

StackDump(L);

MessageBox(NULL,s1,"caption",MB_OK);

lua_pop(lua,1);      // 清除栈里的这个字串

StackDump(L);

lua_pushlstring(L,"MsgOK!",6);  // 把返回值压进栈里

 

// 这个返回是指返回值的个数

return 1;

}

 

       然后就导出这个函数,如下:

       lua_pushcfunction(lua,Msg);

       lua_setglobal(lua,"Msg");

 

       接着就执行刚才的lua文件就行了,记得执行之前要先lua_open () 哦:

       luaL_dofile(lua,"test.lua");

 

       运行的结果就是连续跳出两个messagebox,第一个是123,第二个是"MsgOK!",说明我们返回的字串被lua接收到了,lua的第二行我们没有接收它的返回值,则这个返回值会自动被抛弃了。

       如果需要多返回值,则我们要把下面一句:

lua_pushlstring(L,6);  // 把返回值压进栈里

// 这个返回是指返回值的个数

return 1;

 

改为:

lua_pushlstring(L,6);  // 把返回值压进栈里

lua_pushlstring(L,"haha!",5);      // 把返回值压进栈里

// 这个返回是指返回值的个数

return 2;

 

这样我们在lua文件里就可以像下面一样取得两个返回值了:

c,d = Msg("123");

那c和d就分别是"MsgOK!"和"haha!"两个字串了。 这种自动机制用起来还是比较方便的。

 

3、交互栈

 

上面两个调用其实都是对lua栈的实用,那我们就要好好理解一个概念,lua和vc的交互栈(栈是什么?请参考数据结构的书哈。。。)lua和vc就是通过这个栈来实现交互的,这个栈的访问函数有lua_gettop,lua_settop,lua_tostring,lua_toXXX等等的函数,我们要清楚当一个函数调用发生的时候,栈里是发生了什么。上面我用了一个StackDump函数,当我们调用的时候,能很清楚的看到栈里发生了什么。

       首先我们要知道从栈顶往下数就是-1、-2,从栈底往上数就是1、2。

如果使用lua_gettop(L,1),就是取得栈底第一个元素。lua_gettop(L,-1)就是取得栈顶的第一个元素。lua_pop() (L,1)就是把栈顶的一个元素弹出来,lua_pop()(L,2)就是把栈顶的两个元素弹出。

 

好了,写了一通,最后是这个StackDump函数的实现:

int StackDump(lua_State* L)

{

       int nTop = lua_gettop(L); //得到栈的元素个数。栈顶的位置。

       OutputDebugString("The Length of stack is %d/n",nTop); //输出栈顶位置

       for (int i = 1; i <= nTop; ++i)

       {

              int t = lua_type(L,i);

              OutputDebugString("%s:",lua_typename(L,t)); //这里的typename是把类型的枚举变成字符串,是类型名。不是栈中的位置。

              switch(t)

              {

              case LUA_TNUMBER:

                     OutputDebugString("%f",lua_tonumber(L,i));

                     break;

              case LUA_TSTRING:

                     OutputDebugString("%s",lua_tostring(L,i));

                     break;

              case LUA_TTABLE:

                     //OutputDebugString("%s/n",i));

                     break;

              case LUA_TFUNCTION:

                     //OutputDebugString("%s/n",i));

                     break;

              case LUA_TNIL:

                     OutputDebugString("Is NULL");

                     break;

              case LUA_TBOOLEAN:

                     OutputDebugString("%s",lua_toboolean(L,i) ? "true" : "false");

                     break;

              default:

                     break;

              }

              OutputDebugString("/n");

       }

       return 0;

}

 

本篇文章主要讲了lua和VC的整合、把LUA源代码和VC工程一起编译,VC调用LUA的代码,LUA调用VC的代码,返回值以及多个返回值、交互栈、输出交互栈里的元素信息等内容,下一篇将会说说如何避免阻塞的脚本,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