Lua的栈及基本栈操作
Lua的栈及基本栈操作
https://blog.csdn.net/mydriverc2/article/details/51134737
https://blog.csdn.net/mydriverc2/article/details/51134810
理解Lua栈
Lua通过一个“虚拟栈”与C/C++程序进行数据交互,所有的Lua C API都是通过操作这个栈来完成相应的数据通信。
Lua的这个“虚拟栈”解决了C/C++程序与Lua程序通信的两大问题:
Lua使用垃圾回收,而C/C++需要手动管理内存。
Lua使用动态类型,而C/C++使用的是静态类型。
因 为这个栈在Lua虚拟机内部,当一个Lua的变量放在栈里面的时候,虚拟机可以知道它有没有被宿主程序所使用,从而决定是否采用GC。另外Lua采用结构 体封装了类似“Lua_Value”的类型,让它可以存储任何C的类型。从而在数据交换的时候,任何类型都可以被放入栈的一个slot中。
由于栈是FILO的,所以,当我们在Lua里面操作这个栈的时候,每次操作的都是栈的顶部。而Lua的C API则有更多的控制权,它可以非常灵活地操纵这个栈的任意位置的元素。
基本Lua栈操作
往栈里面压入一个值
|
1
2
3
4
5
6
7
|
void lua_pushnil (lua_State *L);void lua_pushboolean (lua_State *L, int bool);void lua_pushnumber (lua_State *L, lua_Number n);void lua_pushinteger (lua_State *L, lua_Integer n);void lua_pushunsigned (lua_State *L, lua_Unsigned n);void lua_pushlstring (lua_State *L, const char *s, size_t len);void lua_pushstring (lua_State *L, const char *s); |
查询栈里面的元素
|
1
|
int lua_is* (lua_State * L, int index); |
获取栈内给定位置的元素值
|
1
|
xxx lua_toXXX(lua_State * L, int index); |
这里面的xxx可以是nil, boolean, string,integer等等。
其它栈操作
|
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
|
//取得栈中元素个数 int lua_gettop (lua_State *L);//设置栈的大小为一个指定的值,而lua_settop(L,0)会把当前栈清空//如果指定的index大于之前栈的大小,那么空余的空间会被nil填充//如果index小于之前的栈中元素个数,则多余的元素会被丢弃 void lua_settop (lua_State *L, int index);//把栈中index所在位置的元素压入栈 void lua_pushvalue (lua_State *L, int index);//移除栈中index所在位置的元素void lua_remove(lua_State *L, int index);//在栈的顶部的元素移动至index处void lua_insert(lua_State *L, int index);//从栈顶弹出一个值,并把它设置到给定的index处void lua_replace(lua_State *L, int index);//把fromidx处的元素copy一份插入到toidx,这操作不会修改fromidx处的元素void lua_copy(lua_State *L, int fromidx, int toidx); |
另外,根据《Programming In Lua》一书中的所讲,我们可以定义一个函数stackDump来打印当前栈的情况:
|
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
|
static void stackDump(lua_State* L){ cout<<"\nbegin dump lua stack"<<endl; int i = 0; int top = lua_gettop(L); for (i = 1; i <= top; ++i) { int t = lua_type(L, i); switch (t) { case LUA_TSTRING: { printf("'%s' ", lua_tostring(L, i)); } break; case LUA_TBOOLEAN: { printf(lua_toboolean(L, i) ? "true " : "false "); }break; case LUA_TNUMBER: { printf("%g ", lua_tonumber(L, i)); } break; default: { printf("%s ", lua_typename(L, t)); } break; } } cout<<"\nend dump lua stack"<<endl;} |
C/C++访问Lua的Table
假设我们的Lua文件中有一个Table为:
|
1
|
me = { name = "zilongshanren", age = 27} |
我们可以通过以下C代码来访问它的元素:
|
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
|
//从Lua里面取得me这个table,并压入栈lua_getglobal(L, "me");if (!lua_istable(L, -1)) { CCLOG("error! me is not a table");}//往栈里面压入一个key:namelua_pushstring(L, "name");//取得-2位置的table,然后把栈顶元素弹出,取出table[name]的值并压入栈lua_gettable(L, -2); //输出栈顶的nameCCLOG("name = %s", lua_tostring(L, -1));stackDump(L);//把栈顶元素弹出去lua_pop(L, 1);//压入另一个key:agelua_pushstring(L, "age");//取出-2位置的table,把table[age]的值压入栈lua_gettable(L, -2);stackDump(L);CCLOG("age = %td", lua_tointeger(L, -1)); |
Lua5.1还引入了一个新方法:
|
1
|
lua_getfield(L, -1, "age"); |
它可以取代:
|
1
2
3
4
|
//压入另一个key:age lua_pushstring(L, "age"); //取出-2位置的table,把table[age]的值压入栈 lua_gettable(L, -2); |
下篇文章,我们将介绍Lua如何调用C/C++里面的函数。
本篇文章主要介绍C++和Lua相互传递数据。如果你还不知道怎么在C/C++里面调用Lua脚本的话,请参考这篇文章。本文主要介绍基本数据类型的传递,比如整型(int),字符串(string)、数字(number)及bool值。
加载并运行Lua脚本
由于在上一个教程里面已经介绍过如何在C/C++里面嵌入Lua,所以这一节就简单的介绍一下程序怎么用,配置就略过啦。
创建Lua虚拟机
|
1
|
lua_State *lua_state = luaL_newstate(); |
加载Lua库
|
1
2
3
4
5
6
7
8
9
10
11
12
|
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_settop(lua_state, 0); } |
运行Lua脚本
|
1
2
3
4
5
6
7
8
9
10
11
12
|
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只是一个宏定义:
|
1
2
|
#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函数:
|
1
2
3
4
|
-- add two numbersfunction add ( x, y ) return x + yend |
Lua的函数定义是以function为keyword,然后以end结尾,同时它的参数是没有形参类型的,另外,Lua的函数可以返回多个值。不过我们这里只返回了一个值。
接下来,让我们看看如果在C++程序里面调用这个函数:
|
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
|
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;} |
然后,我们就可以在程序里面调用它了:
|
1
|
std::cout<< "2 + 1= " << luaAdd(lua_state,4,1)<<std::endl; |
注意,这个方法调用要在lua_pcall调用之后。
操作Lua全局变量
C++里面获取Lua全局变量的值
首先,我们在hello.lua里面定义一个全局变量
|
1
|
myname = "子龙山人" |
然后我们在C++里面访问它:
|
1
2
3
4
|
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:
|
1
2
|
lua_pushstring(lua_state, "World"); lua_setglobal(lua_state, "myname"); |
这时,我们只要在hello.lua的最开始部分,调用print(myname)就可以打印传递进来的值了。
C++传递Table给Lua
|
1
2
3
4
5
6
7
8
9
10
|
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:
|
1
2
3
|
for i=1,#arg do print(" ", i, arg[i])end |
这里的#arg是获得table的长度,然后使用arg[i]来获取table的索引i处的value。
Lua返回多个值给C++
首先是Lua代码:
|
1
2
3
4
|
local temp = {9, "hehehej"}-- temp[1]=9-- temp[2]="See you space cowboy!"return temp,9,1 |
然后是C++代码:
|
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
|
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; } |
最后输出结果为:
|
1
2
3
4
5
|
[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++里面是倒过来的。因为我们是使用栈作为数据结构来传递数据,而栈是先进后出的。
=========== End
Lua的栈及基本栈操作的更多相关文章
- 用LinkedList集合演示栈和队列的操作
在数据结构中,栈和队列是两种重要的线性数据结构.它们的主要不同在于:栈中存储的元素,是先进后出:队列中存储的元素是先进先出.我们接下来通过LinkedList集合来演示栈和队列的操作. import ...
- C++中栈结构建立和操作
什么是栈结构 栈结构是从数据的运算来分类的,也就是说栈结构具有特殊的运算规则,即:后进先出. 我们可以把栈理解成一个大仓库,放在仓库门口(栈顶)的货物会优先被取出,然后再取出里面的货物. 而从数据的逻 ...
- C++学习---栈的构建及操作
一.顺序栈 #include <iostream> using namespace std; #define MAXSIZE 100 //栈的最大容量 typedef struct { i ...
- C语言描述栈的实现及操作(链表实现)
#include<stdio.h> #include<malloc.h> #include<stdlib.h> typedef int Elementtype; / ...
- Lua与C交互之基础操作(1)
@(语言) Lua是一个嵌入式的语言,可以Lua可以作为程序库用来扩展应用的功能,也可以注册有其他语言实现的函数,这些函数可能由C语言(或其他语言)实现,可以增加一些不容易由Lua实现的功能.这就是L ...
- 写一个栈,实现出栈、入栈、求最小值,时间复杂度为O(1)
#-*-coding:utf-8-*- ''' 需求:写一个栈,实现出栈.入栈.求最小值,时间复杂度为O(1) 思路:通过两个栈实现,一个栈stack,一个辅助栈min_stack,记录stack中的 ...
- 栈(顺序栈)----C语言
栈 栈是一种运算受限的线性表,是一种先进后出的数据结构,限定只能在一端进行插入和删除操作,允许操作的一端称为栈顶,不允许操作的称为栈底 顺序栈(顺序结构) 顺序栈:用一段连续的存储空间来存储栈中的数据 ...
- Java栈之顺序栈存储结构实现
一.栈的基本定义 栈是一种数据结构,它代表一种特殊的线性表,这种线性表只能在固定一端(通常认为是线性表的尾端)进行插入.删除操作的特殊线性表,通常就是在线性表的尾端进行插入.删除操作. 二.顺序栈的实 ...
- Linux虚拟地址空间布局以及进程栈和线程栈总结【转】
转自:http://www.cnblogs.com/xzzzh/p/6596982.html 原文链接:http://blog.csdn.net/freeelinux/article/details/ ...
随机推荐
- CLR 虚方法调用和接口方法调用
不知接口方法和虚方法分发有什么区别?似乎在CIL中都是callvirt指令. 对,MSIL里都是callvirt,但JIT的时候得到了不同的处理:对虚方法的分发是编译成这样: mov ecx, es ...
- Npoi Web 项目中(XSSFWorkbook) 导出出现无法访问已关闭的流
NPOI生产.xlsx文件件时,在使用book.Write(ms);后,会关闭流,这样导致再次使用Respons输出流的时候就出错了. 造成关闭流的主要原因有时其实是跨域,同域是没有问题的. //新建 ...
- Markdown 图片的简单处理
0. 前言 最近写 md 文章的时候发现,在 markdown 里插入一些很长的图片的时候,会显得很不好看,于是去查了一下如何实现 markdown 里图片的并排显示,参考了各个博客内的内容和 mar ...
- 使用Eclipse对SpringBoot项目如何进行打包部署
1,打包概要介绍: 自己做了个小demo,突然想练一下如何打包发布,期间出现了两个错误,第一个是加载不到主类,第二个是加载不到jsp文件,一会会把这两个问题一一陈述,以及解决方法. 1.1,先检查po ...
- nginx使用场景
1. 对外开放本地封闭Server 本地server无法对外开放,nginx做反向代理,对外开发,使得外部可以访问封闭服务. upstream npm { server ; keepalive ; } ...
- Jenkins之自动发送git变更到微信
当我们通过Jenkins构建job的时候,是可以获取到git Change Log 的信息, 即本次上线修改了什么功能,我们将这个信息发送到微信群相关人员可直接获取到上线变更信息, 这样就不需要人为的 ...
- Spring Aop(十二)——编程式的创建Aop代理之AspectjProxyFactory
转发地址:https://www.iteye.com/blog/elim-2397922 编程式的创建Aop代理之AspectjProxyFactory 之前已经介绍了一款编程式的创建Aop代理的工厂 ...
- leetcode 区间合并
个区间若能合并,则第一个区间的右端点一定不小于第二个区间的左端点.所以先把区间集合按照左端点从小到大进行排序,接着从第一个区间开始遍历,对每个区间执行如下操作: 1.首先保存该区间的左端点start和 ...
- python进阶--多线程多进程
一.线程和进程 进程是拥有独立内存,能够独立运行的最小单位,也是程序执行的最小单位,线程是程序运行过程中,一个单一的顺序控制流程,是程序执行流的最小单位,一个进程至少包含一个线程,多线程共享进程的内存 ...
- 大数据(bigdata)练习题
1.在HDFS文件系统的根目录下创建递归目录“1daoyun/file”,将附件中的BigDataSkills.txt文件,上传到1daoyun/file目录中,使用相关命令查看文件系统中1daoyu ...