如何在Lua与C/C++之间实现table数据的交换

为什么80%的码农都做不了架构师?>>> hot3.png

    之前在《C/C++和Lua是如何进行通信的?》一文中简单的介绍了lua与宿主之间的通信。简单的说两种不同的语言之间数据类型不一样又如何进行数据交换呢?那就是lua_State虚拟栈,通过栈操作和lua库函数,我们很轻松就能完成两者之间的数据交换。

    开始之前,明确几个问题,lua中的虚拟栈的索引编号问题(我们假设栈大小为n),编号1是栈底,n视栈顶,编号-1是栈顶,-n是栈底。lua中的库函数需要访问和操作栈上的数据都是通过索引编号定位的。但是我们需要明确一点,有些API并没有使用索引编号作为参数,意味着默认对栈顶进行操作。如lua_pushnumber(L, 66)将数值66压入栈顶,lua_tonumber(L, -1)取编号-1(栈顶)元素等等,如果这些基本知识和API的都已经熟悉了,那么lua与宿主之间的数据交换就很容易理解了。

    姿势准备好了,那么问题来了。需求:我们现在要设计一个UI界面,我们希望这个UI是可以重用的。为了满足这个需求,显然我们必须将UI界面与显示数据分离。使用lua初始化数据后,将数据传递给UI界面然后显示。这样如果需求变更(游戏开发中经常产生这样的需求),我们也只需改变lua脚本就能重用UI界面,听上去真是程序猿的福音啊~~

    用于显示UI的数据必定很多,需要使用lua中的table来封装这些数据,现在给定如下lua table数据:

local tTest = 
{gdp = 1234,info = "this is test about exchange table data!",task = {12, 23, 34, 45},
};

我们的脚本将调用一个程序封装好的c API(TestTable函数),然后将tTest作为参数,压入虚拟栈中,如下:

local tRet = TestTable(tTest);

虽然tTest table已经传给了程序,我们还需要对TestTable这个c API进行定制,使它能够正确的理解这个table中的数据,实现代码如下(LuaTestTable函数类型是lua_CFuntion类型,注册到lua虚拟机中的函数名为TestTable):

int LuaTestTable(lua_State* L)
{printf("stack size = %d\n", lua_gettop(L)); //打印栈中元素的个数lua_pushstring(L, "gdp");            //将gdp字符串压入栈顶//根据栈顶的key获取table中的value,将key(这里的“gdp”)移除,再将value压入栈顶lua_gettable(L, 1);                      printf("%s\n", lua_tostring(L, -1)); //取栈顶元素(注意这里的整型值都是string类型)lua_pop(L, 1); //取完之后清理栈顶printf("stack size = %d\n", lua_gettop(L)); //打印栈中元素的个数lua_pushstring(L, "info");   //同上lua_gettable(L, 1);printf("%s\n", lua_tostring(L, -1));lua_pop(L, 1);printf("stack size = %d\n", lua_gettop(L));lua_pushstring(L, "task"); //这里的value值是一个table哦,没关系栈操作都是一样的lua_gettable(L, 1);for (int i = 0; i < 4; ++i){lua_pushnumber(L, i+1);lua_gettable(L, -2);printf("%s\n", lua_tostring(L, -1));lua_pop(L, 1);}lua_pop(L, 1);printf("stack size = %d\n", lua_gettop(L)); //到这里tTest表依然在栈底,但不影响后面的操作。//------华丽的分割线------------////到这里table数据的解析就结束了,以下内容是c API给lua返回table数据lua_newtable(L);//要给lua脚本返回一个table类型,先要new一个,压入栈顶lua_pushnumber(L, 1); //将key先压入栈lua_pushstring(L, "table2lua"); //再将value压入栈lua_settable(L, -3);//settable将操作-2,-1编号的键值对,设置到table中,并把key-value从栈中移除lua_pushstring(L, "key"); //同上lua_newtable(L); //这里有个子tablelua_pushstring(L, "capi");//这里的value类型使用lua_CFunction类型,可用做c API调用,函数实现请参看附录1lua_pushcfunction(L, LuaSayHello); lua_settable(L, -3);lua_pushnumber(L, 2);lua_pushnumber(L, 10086);lua_settable(L, -3);lua_settable(L, -3); //这个从这里“lua_pushstring(L, "key"); //同上”开始匹配的printf("stack size = %d\n", lua_gettop(L));return 1; //返回栈顶1个元素
}

需要说明的是在lua中tTest["gdp"]和tTest.gdp的调用形式是一样的,这是lua的语法糖。当然操作栈中的table方法除了lua_gettable和lua_settable还有其它方法,请参看lua_rawget和lua_rawset。理解栈中的元素变化是非常重要的。

    LuaTestTable函数API的后面部分介绍了构造一个任意table作为返回值,返回给lua脚本。首先使用lua_newtable库函数新建一个table类型的数据,并压入栈。然后将键值对key-value依次压入栈,调用lua_settable(L, index)将key-value设置到table中,子table操作也是一样的(这里的index指的是要设置的table在栈中的索引编号)。

    好了,基本介绍完了,最后来编写脚本,看看效果(以下是程序调用的脚本):

--file: test.lua
local tTest = {gdp = 1234,info = "this is test about exchange table data!",task = {12, 23, 34, 45},
};
local tRet = TestTable(tTest);
printTable(tRet);    //实现请参看附录2
tRet.key.capi(); //实现请参看附录1

TestTable成功解析tTest的数据,并且返回一个table类型的tRet。

printTable(tRet);

printTable简单的实现了一个table打印的脚本将tRet打印输出。

tRet.key.capi();

脚本调用tRet中返回的c API类型的函数。具体实现请参见附录。

运行结果:

stack size = 1  //以下是LuaTestTable的输出
1234
stack size = 1
this is test about exchange table data!
stack size = 1
12
23
34
45
stack size = 1   
stack size = 2   //以下是printTable的输出
{[1] = table2lua[key] = {[2] = 10086[capi] = function: 0x409795}
}
Lua call c/c++:SayHello() //这里是 [capi] = function: 0x409795被调用的输出
Hello Everyone!

综上述,我们只需要修改tTest中的数据(结构不能改,改了需要修改LuaTestTable函数)就能改变UI界面的显示,成功的解决了UI界面的复用问题。(文中没有具体讲到如何UI截面相关的细节,请参照例子自行脑补。)

附录1:

//注册到lua虚拟机中的c API函数
int LuaSayHello(lua_State* L)
{printf("Lua call c/c++:SayHello()\n");printf("Hello Everyone!\n");return 0;
}

附录2:

//脚本函数实现打印一个table
function printTable(t, n)if "table" ~= type(t) thenreturn 0;endn = n or 0;local str_space = "";for i = 1, n dostr_space = str_space.."  ";endprint(str_space.."{");for k, v in pairs(t) dolocal str_k_v = str_space.."  ["..tostring(k).."] = ";if "table" == type(v) thenprint(str_k_v);printTable(v, n + 1);elsestr_k_v = str_k_v..tostring(v);print(str_k_v);endendprint(str_space.."}");
end

完整项目托管在github上: https://github.com/xlplbo/Lua2Game.git(支持vs2013和cmake编译)

转载请申明出处,如有任何疑问或建议指出,谢谢~

转载于:https://my.oschina.net/xlplbo/blog/398792

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/289202.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

华为ar2200路由器 策略路由_路由交换篇 06 华为路由器RIP协议的配置

任务2 RIP路由协议的配置■ 任务背景某大型企业已有12台路由器&#xff0c;需要组建内部网络实现资源共享&#xff0c;考虑公司规模较大不适合采用静态路由组网&#xff0c;工程师对公司的实际情况进行了分析&#xff1a;1、路由跳数较多&#xff0c;不适合使用静态路由组网&am…

c均值算法的设计与实现_如何使用C链表实现 LRU 算法

什么是 LRU 算法LRU 是一种缓存淘汰策略。计算机的缓存容量有限&#xff0c;如果缓存满了就要删除一些内容&#xff0c;给新的内容腾位置。但是要删除哪些内容呢&#xff1f;我们肯定希望删掉那些没有用的缓存&#xff0c;而把有用的数据继续留在缓存中&#xff0c;方便之后继续…

React Native之编译提示Only one default export allowed per module.

1 问题 部分代码如下 class HomeScreen extends React.Component {render() {return (<View style{{ flex: 1, alignItems: center, justifyContent: center }}><Text>Home Screen</Text></View>);} }export default createStackNavigator({Home: {s…

Matlab插值方法大全

命令1 interp1 功能 一维数据插值(表格查找)。该命令对数据点之间计算内插值。它找出一元函数f(x)在中间点的数值。其中函数f(x)由所给数据决定。 x:原始数据点 Y:原始数据点 xi:插值点 Yi:插值点 格式 (1)yi = interp1(x,Y,xi) 返回插值向量yi,每一元素对应于参量xi,同…

VS Code部署Teams webhook到Azure Functions

点击上方蓝字关注我们&#xff08;本文阅读时间&#xff1a;6分钟&#xff09;Microsoft Teams这款产品对于我们来说已经很熟悉了&#xff0c;作为开发者&#xff0c;我们也可以通过官方的一些开发模式来build我们自己的Teams应用。今天快速跟大家分享一下&#xff0c;如何在VS…

记一次TCP连接异常故障解决

为什么80%的码农都做不了架构师&#xff1f;>>> 一.情况表现为 1.在公司内网对站点的http访问&#xff1a; linux主机出现故障&#xff1a;curl以及抓包分析&#xff0c;发现服务端不响应linux客户端的请求&#xff0c;无法建立TCP连接&#xff0c;浏览器返回“无法…

【转】基于 Android NDK 的学习之旅-----数据传输(引用数据类型)

原文网址&#xff1a;http://www.cnblogs.com/luxiaofeng54/archive/2011/08/20/2147086.html 基于 Android NDK 的学习之旅-----数据传输二&#xff08;引用数据类型&#xff09;(附源码) 基于 Android NDK 的学习之旅-----数据传输&#xff08;引用数据类型&#xff09; 接着…

微软官方pe工具_微软官方下载工具

二、进入官网下载百度搜索“win10下载”进入微软官网win10下载地址&#xff0c;点击“立即下载工具”开始下载&#xff1b;三、运行软件下载完成后&#xff0c;右键以“管理员身份”打开&#xff0c;点击“接受”。四、选择用途选择第一个升级自己的电脑&#xff1b;选择第二个…

编程之美3——N!末尾有多少个0

因为n!是一个非常大的数&#xff0c;所以不能通过常规的方法&#xff0c;求出n!的值之后&#xff0c;再判断它的末尾有多少个0.这里的关键就是&#xff0c;n!的末尾有多少个0&#xff0c;取决于n!中&#xff0c;质因数 5 的个数。如&#xff1a;12!479001600 &#xff0c;其…

Git之添加公钥之后git clone ****提示sign_and_send_pubkey: signing failed: agent refused operation

1 问题 在GitlLab上面添加了公钥之后,然后执行git clone *****提示下面的错误 sign_and_send_pubkey: signing failed: agent refused operation 2 解决办法 执行下面2个命令即可 eval "$(ssh-agent -s)" ssh-add ssh-agent是一种控制用来保存公钥身份验证所使…

mysql老是自动停止_ecs云服务器 mysql经常自动停止挂掉重启问题分析

我的ecs服务器为1g内存的配置&#xff0c;在部署了nginx,mysql,redis,node服务后跑起项目来&#xff0c;(mysql使用默认配置),每过几天便发现了经常会出现数据库自动停止挂掉&#xff0c;然后几分钟后重启的现象&#xff0c;与此同时ecs无法登陆,当然&#xff0c;网站也是无法访…

C#遍历指定文件夹中的所有文件

C#遍历指定文件夹中的所有文件 DirectoryInfo TheFolder=new DirectoryInfo(folderFullName); //遍历文件夹 foreach(DirectoryInfo NextFolder in TheFolder.GetDirectories()) this.listBox1.Items.Add(NextFolder.Name); //遍历文件 foreach(FileInfo NextFile in TheFo…

Task.Factory.StartNewTResult 和 Task.RunTResult 到底有什么区别?

前言这不是和《Task.Factory.StartNew 和 Task.Run 到底有什么区别&#xff1f;》一样吗&#xff0c;怎么又写一篇&#xff1f;起先我也是这么觉得的&#xff0c;但实际发现并非如此。实现代码查看这 2 个方法的内部实现&#xff0c;其内部实现逻辑其实是一样的&#xff0c;只是…

大数据服务社会的一个有益实践

自从有了微信&#xff0c;很多人的生活习惯就发生了不同以往的变化。比如&#xff0c;早上起床后的第一件事&#xff0c;可能就是把积压的未读微信信息快速浏览一遍&#xff0c;该发的发、该转的转。笔者虽离开工作岗位&#xff0c;却依然总有时间不够用的感觉&#xff0c;所以…

python pysnmp使用

SNMP标准引入一组ASN.1语言元素&#xff0c;称之为SMI&#xff08;Structure of Management Information&#xff09;。由SMI描述的相互关联的被管对象&#xff08;Managed Objects&#xff09;组成MIB&#xff08;Management Information Base&#xff09;模块。核心MIB中经常…

Git之checkout到别的分支提示Your local changes to the following files would be overwritten by checkout:

1 问题 在我自己的分支,然后切换到主分支,提示错误如下 Your local changes to the following files would be overwritten by checkout:****file****file 2 解决办法 先把这些文件进行add操作,然后再进行commit,就可以了, git add filegit commit -m commit message 然后你…

javaweb连接不上mysql怎么办_java web应用连接mysql会突然connection连接失败

tomcat6.0mysql5.1项目&#xff1a;java web项目问题&#xff1a;原本项目运行了好几天了&#xff0c;一直没发现问题&#xff0c;突然今天报数据库连接异常&#xff0c;进入看日志发现### Error querying database. Cause: com.mysql.jdbc.exceptions.jdbc4.MySQLNonTransien…

【遥感物候】C#遥感数据GIMMS 3G NDVI头文件批量生成器(几何校正)

长时间序列全球NDVI数据GIMMS 3g(点击下载),原数据无投影,格式为VI3g,IE浏览器下载的为.txt格式。GIS软件无法直接打开,Envi 5可以打开。 ENVI中点击File->Open File as->Binary会弹出填写元数据的窗口,填写基本信息: Envi5.1中打开二进制(Binary)数据是,需要…

Kuzzle,一种内部部署的文档后端

Kuzzle是一种可以内部部署或是在云中运行的文档后端。在近期的CES 2017上&#xff0c;提供该平台的公司公布了其企业版解决方案。 Kuzzle用NoSQL仓库对文档做持久保存&#xff0c;支持基于模式的或是无模式的文档。Kuzzle提供CRUD API&#xff0c;并使用了Elasticsearch提供高级…

分享一个基于.NET6包含DDD,ES,CQRS等概念的开源项目

当你在学习DDD、CQRS或时间溯源时&#xff0c;除了大量的学习资源&#xff08;比如书籍和文章&#xff09;之外&#xff0c;你还接触到了许多概念&#xff0c;这些资源只是在讨论理论问题。这很好&#xff0c;我们知道他们在说什么&#xff0c;但我们如何在一个真正的项目中使用…