Lua C Api lua_gettable 、lua_settable 、lua_next 使用详解
之前一直没理清lua_gettable和lua_settable的使用,今天理清了,顺便就做下笔记了。
1.lua_gettable void lua_gettable (lua_State *L, int index);
把t[k]
值压入堆栈,这里的 t
是指有效索引 index
指向的值,而 k
则是栈顶放的值。这个函数会弹出堆栈上的 key,把结果放在栈上相同位置。
下面举个例子:
// 将一个key放到栈顶,这个key为1。如果你的key是字符串,那就用lua_pushstring。 lua_pushnumber(L, ); // table一开始是在栈顶,即-1处的,但上面的语句压入了一个值,栈顶变-2了。 // lua_gettable的作用就是以栈顶的值作为key来访问-2位置上的table。 lua_gettable(L, -);
这时table中的第1个元素的值就放到栈顶了,你想怎么使用就怎么使用吧。
获取table元素:
* 将元素的key压入到栈中,用 lua_gettable(Lua_state,index)
* 对于字符串索引,可以用lua_getfield(Lua_state,index,key)来直接获取,
如:lua_getfield(stack, -1, "loaded");等价于 lua_pushstring(L,"loaded") lua_gettable(L,-2);
上面说的是访问table中的一个元素的方法,那要怎么样遍历table中的所有元素呢?
1)如果table是一个以连续的整形作为key的table, 可以用下面方法:
int size = lua_objlen(L,-);//相关于#table
for(int i = ; i <= size; i++)
{
lua_pushnumber(L, i);
lua_gettable(L, -);
//这时table[i]的值在栈顶了
lua_pop(L, );//把栈顶的值移出栈,保证栈顶是table以便遍历。
};
2)如果table中的key是任意值呢?可以用下面的方法:
lua_pushnill(L);
while(lua_next(L, -))
{
//这时值在-1(栈顶)处,key在-2处。
lua_pop(L, );//把栈顶的值移出栈,让key成为栈顶以便继续遍历
}
这里重点说明一下lua_next。
它执行操作是这样的,以栈顶元素为key,先判断上一个key的值(这个值放在栈顶,如果是nil,则表示当前取出的是table中第一个元素的值),然后得到当前的key和value。
这时先把栈顶出栈,将新key进栈,后将value进栈。这样栈顶就是table中第一个遍历到的元素的值。用完这个值后,我们要把这个值出栈,让新key在栈顶以便继续遍历。当根据上一个key值算不出下一个key值时(其实这时候key的是多少并不重要,只要不为nil就行,因为为nil会返回table的第一个元素),lua_next返回0,结束循环。
2.lua_settable
void lua_settable (lua_State *L, int index);
作一个等价于 t[k] = v
的操作, 这里 t
是一个给定有效索引 index
处的值, v
指栈顶的值, 而 k
是栈顶之下的那个值。
这个函数会把键和值都从堆栈中弹出。
其实这个解释的意思就是,lua_settable 会把栈顶作为value,栈顶的下一个作为key设置到index指向的table,最后把这两个弹出弹出栈,这时候settable完成。 3.lua_next
int lua_next (lua_State *L, int index);
从栈上弹出一个 key(键),然后把索引指定的表中 key-value(健值)对压入堆栈(指定 key 后面的下一 (next) 对)。如果表中以无更多元素,那么lua_next
将返回 0 (什么也不压入堆栈)。
典型的遍历方法是这样的:
/* table 放在索引 't' 处 */
lua_pushnil(L); /* 第一个 key */
while (lua_next(L, t) != ) {
/* 用一下 'key' (在索引 -2 处) 和 'value' (在索引 -1 处) */
printf("%s - %s\n",
lua_typename(L, lua_type(L, -)),
lua_typename(L, lua_type(L, -)));
/* 移除 'value' ;保留 'key' 做下一次迭代 */
lua_pop(L, );
}
在遍历一张表的时候,不要直接对 key 调用 lua_tolstring
,除非你知道这个 key 一定是一个字符串。调用 lua_tolstring
有可能改变给定索引位置的值;这会对下一次调用 lua_next
造成影响。
最后我们以一段c调用lua的loaded表并且设置符合某个规章的key将它的value设置为nil,为列:
ua_getglobal(stack, "package"); /* L: package,获得package,在栈定 */
lua_getfield(stack, -, "loaded"); /* L: package loaded,获得表,在栈顶*/
lua_pushnil(stack); /* L: package loaded nil */
while ( != lua_next(stack, - ) ) /* L: package loaded, key, value,上一个栈顶为nil,弹出nil,获得表的第一个key和value,压入栈 */
{
//CCLOG("%s - %s \n", tolua_tostring(stack, -2, ""), lua_typename(stack, lua_type(stack, -1)));
std::string key=tolua_tostring(stack, -, ""); /*这时候栈顶得下一个,是key*/
std::string tableKey =key; /*下面是对key的一段处理*/
int found = tableKey.rfind(".lua");
if (found!=std::string::npos)
tableKey = tableKey.substr(,found);
tableKey=replaceAll(tableKey,".","/");
tableKey=replaceAll(tableKey,"\\","/");
tableKey.append(".lua");
found = fileName.rfind(tableKey);
if ( == found || ( found!=std::string::npos && fileName.at(found-) == '/'))
{
lua_pushstring(stack, key.c_str()); /*package loaded, key, value,newkey, 将key,压入栈顶*/
lua_pushnil(stack); /* pakage,loaded(table)(-5),key(-4),value(-3),key(-2),nil(-1)*/
if (lua_istable(stack, -)) /*判读栈顶往下第五个是不是table*/
{
/*结果将key对应的值置为nil*/
lua_settable(stack, -);/*pakage,loaded(table),key,value, 将栈顶两个元素作为key和value设置给table,弹出栈顶两个元素*/
}
}
lua_pop(stack, ); /*pakage,loaded(table),key 弹出value,留下key作为下一个next*/
}
lua_pop(stack, ); /*栈平衡*/
Lua C Api lua_gettable 、lua_settable 、lua_next 使用详解的更多相关文章
- Lua中强大的元方法__index详解
今天要来介绍比较好玩的内容:__index元方法 我是备胎,记得回头看看 咳咳,相信每一位女生都拥有或者不知不觉中拥有了一些备胎,啊!当然,又或许是成为过别人的备胎. 没有备胎的人,就不是完整的人生. ...
- Java数据持久层框架 MyBatis之API学习十(Logging详解)
对于MyBatis的学习而言,最好去MyBatis的官方文档:http://www.mybatis.org/mybatis-3/zh/index.html 对于语言的学习而言,马上上手去编程,多多练习 ...
- Lua中的模块与module函数详解
很快就要开始介绍Lua里的“面向对象”了,在此之前,我们先来了解一下Lua的模块. 1.编写一个简单的模块 Lua的模块是什么东西呢?通常我们可以理解为是一个table,这个table里有一些变量.一 ...
- API联调神器PostMan使用详解
简介 创建 + 测试:创建和发送任何的HTTP请求,请求可以保存到历史中再次执行 Organize:使用Postman Collections为更有效的测试及集成工作流管理和组织APIs docume ...
- Queue API的几种实现详解
目录 Queue API的几种方法的使用 ArrayBlockingQueue原理及源码解析 ArrayBlockingQueue的成员变量 ArrayBlockingQueue的offer和put方 ...
- 【地图API】收货地址详解2
上次讲解的方法是: 在地图中心点添加一个标注,每次拖动地图就获取地图中心点,再把标注的位置设置为地图中心点.可参考教程:http://www.cnblogs.com/milkmap/p/6126424 ...
- Lua中ipairs和pairs的区别详解
迭代器for遍历table时,ipairs和pairs的区别: 区别一:ipairs遇到nil会停止,pairs会输出nil值然后继续下去 区别二: , b = , x = , y = , " ...
- Java多线程:实现API接口创建线程方式详解
先看例子: /**实现Runnable接口创建线程步骤: * 1.创建一个实现Runnable接口的类 * 2.重写Runnable类中抽象的run()方法 * 3.创建实现类的对象 * 4.声明Th ...
- restful风格详解
一.概念 RESTful架构,就是目前最流行的一种互联网软件架构.它结构清晰.符合标准.易于理解.扩展方便,所以正得到越来越多网站的采用. REST这个词,是Roy Thomas Fielding在他 ...
随机推荐
- 快学Scala-第四章 映射和元组
知识点: 1.构造映射,映射是对偶的集合 val scores1 = Map("Alice" -> 10, "Bob" -> 7, "Ci ...
- JS的className,字体放大缩小
<!DOCTYPE HTML> <html> <head> <meta http-equiv="Content-Type" content ...
- 代码创建xml文档并写入指定节点
//首先创建 XmlDocument xml文档 XmlDocument xml = new XmlDocument(); //创建根节点 config XmlElement config = xml ...
- hdu 1001 二叉树搜索
Problem Description 判断两序列是否为同一二叉搜索树序列 Input 开始一个数n,(1<=n<=20) 表示有n个需要判断,n= 0 的时候输入结束.接下去一行是一个序 ...
- PHP开源CRM-推荐几个
http://www.xinyou88.com/about/xcrm.html 因为医院要同步用户到诊统计的信息.是拿着表单来回送.途中大概有20分钟左右.有些机器和互联网可以搞定的事情.人力来做.在 ...
- JSON对象长度和遍历方法(转)
最 近在修改一个HTML页面的JS的时候遍历JSON对象,却怎么也调试不通过.怪这个HTML网页不知道用了什么方法禁止了js错误提示,刚开始的时候不 知道有这个问题,用chrome的开发人员工具都没发 ...
- Redis SAVE 命令 返回 ERR
今天使用redis-cli客户端中执行SAVE命令返回 (error) ERR Baidu找不到答案,去Google找一下 应该是redis-server服务没有root权限 然后sudo kill ...
- zabbix Lack of free swap space
Zabbix初始设计是大型公司用于监控服务器集群的,但日常中也用于监控VPS或云主机.后者情况下Zabbix的很多配置和属性就没有经过优化,取决于监控的对象和用途,经常需要对一些Zabbix配置进行调 ...
- ext3文件系统目录限制问题
昨晚排查了在KVM的build系统中的一个问题,跟踪到后面发现在一个目录下mkdir创建目录失败.我手动试了一下,提示如下:cannot create directory `/home/master/ ...
- cakephp , the subquery (2)
Cakephp 框架帮我们做了很多的工作,的确省了我们很多工作,提高了效率. 但是,碰到一些比较复杂的查询时,还是有些问题,官方的cookbook api 有说明一些详细的用法,但感觉还是不太够,有些 ...