欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

Lua教程(二):C++和Lua相互传递数据示例

程序员文章站 2022-03-07 10:20:00
这是我的lua系列教程的第二篇,本篇文章主要介绍c++和lua相互传递数据。如果你还不知道怎么在c/c++里面调用lua脚本的话,请参考. 本文主要介绍基本数据类型的传递,...

这是我的lua系列教程的第二篇,本篇文章主要介绍c++和lua相互传递数据。如果你还不知道怎么在c/c++里面调用lua脚本的话,请参考. 本文主要介绍基本数据类型的传递,比如整形(int),字符串(string)、数字(number)及bool值。

加载并运行lua脚本

由于在上一个教程里面已经介绍过如何在c/c++里面嵌入lua,所以这一节就简单的介绍一下程序怎么用,配置就略过啦。

创建lua虚拟机

复制代码 代码如下:

lua_state *lua_state = lual_newstate();

加载lua库

复制代码 代码如下:

static const lual_reg lualibs[] =
    {
        {"base", luaopen_base},
        {"io", luaopen_io},
        {null, null}
    };
    const lual_reg *lib = lualibs;
    for(; lib->func != null; lib++)
    {
        lual_requiref(lua_state, lib->name, lib->func, 1);
        lua_pop(lua_state,1);
    }

运行lua脚本

复制代码 代码如下:

std::string scriptpath = fileutils::getinstance()->fullpathforfilename("hello.lua");
int status = lual_loadfile(lua_state, scriptpath.c_str());
std::cout << " return: " << status << std::endl;
int result = 0;
if(status == lua_ok)
{
    result = lua_pcall(lua_state, 0, lua_multret, 0);
}
else
{
    std::cout << " could not load the script." << std::endl;
}

这里我们使用的是lual_loadfile而不是之前的lual_dofile,其实lual_dofile只是一个宏定义:

复制代码 代码如下:

#define lual_dofile(l, fn) \
    (lual_loadfile(l, fn) || lua_pcall(l, 0, lua_multret, 0))

我们先调用lual_loadfile可以判断lua脚本是否加载成功,然后再调用lua_pcall来执行lua脚本。

c/c++调用lua函数

首先,我们在hello.lua里面定义一个lua函数:

复制代码 代码如下:

-- add two numbers
function add ( x, y )
    return x + y
end

lua的函数定义是以function为keyword,然后以end结尾,同时它的参数是没有形参类型的,另外,lua的函数可以返回多个值。不过我们这里只返回了一个值。

接下来,让我们看看如果在c++程序里面调用这个函数:

复制代码 代码如下:

int luaadd(lua_state *lua_state , int x, int y)
{
    int sum;
    //获取lua里面的add函数并把它放到lua的栈顶
    lua_getglobal(lua_state, "add");
    //往lua栈里面压入两个参数
    lua_pushnumber(lua_state, x);
    lua_pushnumber(lua_state, y);
    //调用lua函数,这里的2是参数的个数,1是返回值的个数
    lua_call(lua_state, 2, 1);
    //从栈顶读取返回值,注意这里的参数是-1
    sum = lua_tointeger(lua_state, -1);
    //最后我们把返回值从栈顶拿掉
    lua_pop(lua_state, 1);
    return sum;
}

然后,我们就可以在程序里面调用它了:

复制代码 代码如下:

std::cout<< "2 + 1= " << luaadd(lua_state,4,1)<<std::endl;

注意这个方法调用要在lua_pcall调用之后。

操作lua全局变量

c++里面获取lua全局变量的值

首先,我们在hello.lua里面定义一个全局变量

复制代码 代码如下:

myname = "子龙山人"

然后我们在c++里面访问它:

复制代码 代码如下:

lua_getglobal(lua_state, "myname");
std::string myname = lua_tostring(lua_state, -1);
lua_pop(lua_state, 1);
std::cout<<"hello: "<<myname<<std::endl;

这一次我们又是通过lua_getglobal来把myname这个全局变量压到lua栈,然后用lua_tostring来取这个值。

c++里面修改lua全局变量的值

这次我们使用的是lua_setglobal来传递数据给lua:

复制代码 代码如下:

lua_pushstring(lua_state, "world");
    lua_setglobal(lua_state, "myname");

这时,我们只要在hello.lua的最开始部分,调用print(myname)就可以打印传递进来的值了。

c++传递table给lua

复制代码 代码如下:

lua_createtable(lua_state, 2, 0);
    lua_pushnumber(lua_state, 1);
    lua_pushnumber(lua_state, 49);
//    lua_settable(lua_state, -3);
    lua_rawset(lua_state, -3);
    lua_pushnumber(lua_state, 2);
    lua_pushstring(lua_state, "life is a beach");
//    lua_settable(lua_state, -3);
    lua_rawset(lua_state, -3);
    lua_setglobal(lua_state, "arg");

这里我们传递了一个table给lua,这个table为{49,”life is a beach”}。lua table的索引是从1开始的,然后我们在lua脚本里面可以这样子来访问这个table:

复制代码 代码如下:

for i=1,#arg do
    print("      ", i, arg[i])
end

这里的#arg是获得table的长度,然后使用arg[i]来获取table的索引i处的value。

lua返回多个值给c++

首先是lua代码:

复制代码 代码如下:

local temp = {9, "hehehej"}
-- temp[1]=9
-- temp[2]="see you space cowboy!"
return temp,9,1

然后是c++代码:

复制代码 代码如下:

std::stringstream str_buf;
    while(lua_gettop(lua_state))
    {
        str_buf.str(std::string());
        str_buf << " ";
        switch(lua_type(lua_state, lua_gettop(lua_state)))
        {
            case lua_tnumber:
                str_buf << "script returned the number: "
                << lua_tonumber(lua_state, lua_gettop(lua_state));
                break;
            case lua_ttable:
                str_buf << "script returned a table";
                break;
            case lua_tstring:
                str_buf << "script returned the string: "
                << lua_tostring(lua_state, lua_gettop(lua_state));
                break;
            case lua_tboolean:
                str_buf << "script returned the boolean: "
                << lua_toboolean(lua_state, lua_gettop(lua_state));
                break;
            default:
                str_buf << "script returned an unknown-type value";
        }
        lua_pop(lua_state, 1);
        std::cout << str_buf.str() << std::endl;
    }

最后输出结果为:

复制代码 代码如下:

[c++] values returned from the script:
 script returned the number: 1
 script returned the number: 9
 script returned a table
[c++] closing the lua state

在lua里面return值的顺序是table,9,1,而在c++里面是倒过来的。因为我们是使用栈作为数据结构来传递数据,而栈是先进后出的。

下一篇文章,我们将介绍一下c++调用lua的table。