- ubuntu12.04环境下使用kvm ioctl接口实现最简单的虚拟机
- Ubuntu 通过无线网络安装Ubuntu Server启动系统后连接无线网络的方法
- 在Ubuntu上搭建网桥的方法
- ubuntu 虚拟机上网方式及相关配置详解
CFSDN坚持开源创造价值,我们致力于搭建一个资源共享平台,让每一个IT人在这里找到属于你的精彩世界.
这篇CFSDN的博客文章Lua教程(二十一):编写C函数的技巧由作者收集整理,如果你对这篇文章有兴趣,记得点赞哟.
1. 数组操作:
在Lua中,“数组”只是table的一个别名,是指以一种特殊的方法来使用table。出于性能原因,Lua的C API为数组操作提供了专门的函数,如: 。
以上两个函数分别用于读取和设置数组中的元素值。其中index参数表示待操作的table在栈中的位置,key表示元素在table中的索引值。由于这两个函数均为原始操作,比涉及元表的table访问更快。通常而言,作为数组使用的table很少会用到元表.
。
见如下代码示例和关键性注释:
#include <stdio.h> #include <string.h> #include <lua.hpp> #include <lauxlib.h> #include <lualib.h> 。
。
extern "C" int mapFunc(lua_State* L) { //检查Lua调用代码中传递的第一个参数必须是table。否则将引发错误。 luaL_checktype(L,1,LUA_TTABLE); luaL_checktype(L,2,LUA_TFUNCTION); //获取table中的字段数量,即数组的元素数量。 int n = lua_objlen(L,1); //Lua中的数组起始索引习惯为1,而不是C中的0。 for (int i = 1; i <= n; ++i) { lua_pushvalue(L,2); //将Lua参数中的function(第二个参数)的副本压入栈中。 lua_rawgeti(L,1,i); //压入table[i] lua_call(L,1,1); //调用function(table[i]),并将函数结果压入栈中。 lua_rawseti(L,1,i); //table[i] = 函数返回值,同时将返回值弹出栈。 } 。
//无结果返回给Lua代码。 return 0; } 。
。
2. 字符串操作:
当一个C函数从Lua收到一个字符串参数时,必须遵守两条规则:不要在访问字符串时从栈中将其弹出,不要修改字符串。在Lua的C API中主要提供了两个操作Lua字符串的函数,即: 。
第一个API用于截取指定长度的子字符串,同时将其压入栈中。而第二个API则类似于C库中的sprintf函数,并将格式化后的字符串压入栈中。和sprintf的格式说明符不同的是,该函数只支持%%(表示字符%)、%s(表示字符串)、%d(表示整数)、%f(表示Lua中的number)及%c(表示字符)。除此之外,不支持任何例如宽度和精度的选项.
。
。
#include <stdio.h> #include <string.h> #include <lua.hpp> #include <lauxlib.h> #include <lualib.h> 。
。
extern "C" int splitFunc(lua_State* L) { const char* s = luaL_checkstring(L,1); const char* sep = luaL_checkstring(L,2); //分隔符 const char* e; int i = 1; lua_newtable(L); //结果table while ((e = strchr(s,*sep)) != NULL) { lua_pushlstring(L,s,e - s); //压入子字符串。 //将刚刚压入的子字符串设置给table,同时赋值指定的索引值。 lua_rawseti(L,-2,i++); s = e + 1; } //压入最后一个子串 lua_pushstring(L,s); lua_rawseti(L,-2,i); return 1; //返回table。 } 。
。
Lua API中提供了lua_concat函数,其功能类似于Lua中的".."操作符,用于连接(并弹出)栈顶的n个值,然后压入连接后的结果。其原型为: void lua_concat(lua_State *L, int n); 参数n表示栈中待连接的字符串数量。该函数会调用元方法。然而需要说明的是,如果连接的字符串数量较少,该函数可以很好的工作,反之,则会带来性能问题。为此,Lua API提供了另外一组函数专门解决由此而带来的性能问题,见如下代码示例:
#include <stdio.h> #include <string.h> #include <lua.hpp> #include <lauxlib.h> #include <lualib.h> 。
。
extern "C" int strUpperFunc(lua_State* L) { size_t len; luaL_Buffer b; //检查参数第一个参数是否为字符串,同时返回字符串的指针及长度。 const char* s = luaL_checklstring(L,1,&len); //初始化Lua的内部Buffer。 luaL_buffinit(L,&b); //将处理后的字符依次(luaL_addchar)追加到Lua的内部Buffer中。 for (int i = 0; i < len; ++i) luaL_addchar(&b,toupper(s[i])); //将该Buffer及其内容压入栈中。 luaL_pushresult(&b); return 1; } 。
。
使用缓冲机制的第一步是声明一个luaL_Buffer变量,并用luaL_buffinit来初始化它。初始化后,就可通过luaL_addchar将一个字符放入缓冲。除该函数之外,Lua的辅助库还提供了直接添加字符串的函数,如: 。
最后luaL_pushresult会更新缓冲,并将最终的字符串留在栈顶。通过这些函数,就无须再关心缓冲的分配了。但是在追加的过程中,缓冲会将一些中间结果放到栈中。因此,在使用时要留意此细节,只要保证压入和弹出的次数相等既可。Lua API还提供一个比较常用的函数,用于将栈顶的字符串或数字也追加到缓冲区中,函数原型为: 。
3. 在C函数中保存状态: Lua API提供了三种方式来保存非局部变量,即注册表、环境和upvalue。 1). 注册表: 注册表是一个全局的table,只能被C代码访问。通常用于保存多个模块间的共享数据。我们可以通过LUA_REGISTRYINDEX索引值来访问注册表.
。
。
#include <stdio.h> #include <string.h> #include <lua.hpp> #include <lauxlib.h> #include <lualib.h> 。
。
void registryTestFunc(lua_State* L) { lua_pushstring(L,"Hello"); lua_setfield(L,LUA_REGISTRYINDEX,"key1"); lua_getfield(L,LUA_REGISTRYINDEX,"key1"); printf("%s\n",lua_tostring(L,-1)); } 。
int main() { lua_State* L = luaL_newstate(); registryTestFunc(L); lua_close(L); return 0; } 。
。
2). 环境: 如果需要保存一个模块的私有数据,即模块内各函数需要共享的数据,应该使用环境。我们可以通过LUA_ENVIRONINDEX索引值来访问环境。 。
#include <lua.hpp> #include <lauxlib.h> #include <lualib.h> 。
。
//模块内设置环境数据的函数 extern "C" int setValue(lua_State* L) { lua_pushstring(L,"Hello"); lua_setfield(L,LUA_ENVIRONINDEX,"key1"); return 0; } 。
//模块内获取环境数据的函数 extern "C" int getValue(lua_State* L) { lua_getfield(L,LUA_ENVIRONINDEX,"key1"); printf("%s\n",lua_tostring(L,-1)); return 0; } 。
static luaL_Reg myfuncs[] = { {"setValue", setValue}, {"getValue", getValue}, {NULL, NULL} },
extern "C" __declspec(dllexport) int luaopen_testenv(lua_State* L) { lua_newtable(L); //创建一个新的表用于环境 lua_replace(L,LUA_ENVIRONINDEX); //将刚刚创建并压入栈的新表替换为当前模块的环境表。 luaL_register(L,"testenv",myfuncs); return 1; } 。
。
Lua测试代码如下.
3). upvalue: upvalue是和特定函数关联的,我们可以将其简单的理解为函数内的静态变量.
#include <lua.hpp> #include <lauxlib.h> #include <lualib.h> 。
。
extern "C" int counter(lua_State* L) { //获取第一个upvalue的值。 int val = lua_tointeger(L,lua_upvalueindex(1)); //将得到的结果压入栈中。 lua_pushinteger(L,++val); //赋值一份栈顶的数据,以便于后面的替换操作。 lua_pushvalue(L,-1); //该函数将栈顶的数据替换到upvalue(1)中的值。同时将栈顶数据弹出。 lua_replace(L,lua_upvalueindex(1)); //lua_pushinteger(L,++value)中压入的数据仍然保留在栈中并返回给Lua。 return 1; } 。
extern "C" int newCounter(lua_State* L) { //压入一个upvalue的初始值0,该函数必须先于lua_pushcclosure之前调用。 lua_pushinteger(L,0); //压入闭包函数,参数1表示该闭包函数的upvalue数量。该函数返回值,闭包函数始终位于栈顶。 lua_pushcclosure(L,counter,1); return 1; } 。
static luaL_Reg myfuncs[] = { {"counter", counter}, {"newCounter", newCounter}, {NULL, NULL} },
extern "C" __declspec(dllexport) int luaopen_testupvalue(lua_State* L) { luaL_register(L,"testupvalue",myfuncs); return 1; } 。
。
Lua测试代码如下.
。
require "testupvalue" 。
。
func = testupvalue.newCounter(); print(func()); print(func()); print(func()),
func = testupvalue.newCounter(); print(func()); print(func()); print(func()),
--[[ 输出结果为: 1 2 3 1 2 3 --]] 。
。
最后此篇关于Lua教程(二十一):编写C函数的技巧的文章就讲到这里了,如果你想了解更多关于Lua教程(二十一):编写C函数的技巧的内容请搜索CFSDN的文章或继续浏览相关文章,希望大家以后支持我的博客! 。
我是新手。查看 Google 新闻...上下滚动页面时请注意左侧导航栏。 看看它是如何滚动一点,然后在它消失之前粘在页面顶部的? 关于如何做到这一点有什么想法吗? jQuery 和 CSS 可以复制吗
技巧 1:在 Web 服务器上缓存常用数据 技巧 2:在 Application 或 Session 对象中缓存常用数据 技巧 3:在 Web 服务器磁盘上缓存数据和 HTML 技巧 4:避免
我在 excel 中有一个电子表格,其中包含以下行: COLUMN Value1.Value2.Value3 Value4.Value5.Value6 Value7.Value8.Val
GNU Makefile 中是否有任何技巧来获取规则的所有依赖项? 例子: rule1: dep1_1 dep1_2 dep1_3 rule2: dep2_1 dep2_2 rule1 dump_
人们使用什么来追踪内存泄漏?我已经通过代码检查设法解决了一些问题,但我不知道下一步该做什么/当我的程序变大时我将如何管理问题。我知道我在泄漏什么类型的对象,但我不知道是什么让它保持活力。 在 Wind
有什么好的方法可以将“xlSum”、“xlAverage”和“xlCount”等字符串转换为它们在 Microsoft.Office.Interop.Excel.XlConsolidationFunc
我们都见过这个: javascript:document.body.contentEditable='true'; document.designMode='on';无效 0 但我的问题是,这实际上是
我的应用程序将输出一个图形,其布局由用户定义。自定义布局类应该实现我定义的接口(interface)。我应该怎么做?有一个特殊的文件夹,我可以在其中查找布局类?用户是否将类名作为参数传递给应用? 如有
我在弄清楚如何在 Javascript 中自引用表行时遇到了一些麻烦。 这是简化的代码: $( "#listitems tbody" ).append( "" + "" + id.va
关闭。这个问题需要更多focused .它目前不接受答案。 想改进这个问题吗? 更新问题,使其只关注一个问题 editing this post . 关闭 6 年前。 Improve this q
我正在将代码库从一种编程风格转移到另一种编程风格。 我们有一个名为 Operand 的类型,定义如下: class Operand {...}; 然后我们有 class OperandFactory
我使用以下缩略图类在我的内容包装器中显示 4x3 缩略图: .thumbnail { float:left; width:300px; height:200px; ma
按照目前的情况,这个问题不适合我们的问答形式。我们希望答案得到事实、引用或专业知识的支持,但这个问题可能会引发辩论、争论、投票或扩展讨论。如果您觉得这个问题可以改进并可能重新打开,visit the
我认为这是不可能的,但我想在放弃之前问问你。 我想要类似 constexpr 增量的东西。 #include constexpr int inc() { static int inc = 0;
是否有任何适合 C++ 新手的技术和描述的好列表。我在想一个描述 RAII、RVO、左值的列表……这适用于目前不了解这些技术或来自不适用这些技术的其他语言的新手。 最好是短小精悍的:-) 最佳答案 是
我有一个二进制字符串 '01110000',我想在不编写 forloop 的情况下返回前面的前导零数。有谁知道如何做到这一点?如果字符串立即以“1”开头,最好也返回 0 最佳答案 如果您真的确定它是一
我需要优化我的应用程序的 RAM 使用率。 请省去那些告诉我在编写 Python 代码时不应该关心内存的讲座。我有内存问题,因为我使用非常大的默认字典(是的,我也想快点)。我目前的内存消耗是 350M
有时,当我看到一个我喜欢的网站或来自受人尊敬的人的网站时,我会查看源代码并尝试理解它们(就像我们所有人一样)。 关于 Jeremy Keiths他使用以下代码的网站: [role="navigatio
这是我怎样设置 Git 来管理我的家目录的方法。 我有好几台电脑。一台笔记本电脑用于工作,一台工作站放在家里,一台树莓派(或四台),一台 Pocket CHIP,一台 运行
shell 技巧 表变量 HBase 0.95 版本增加了为表提供 jruby 风格的面向对象引用的 shell 命令。以前,作用于表的所有 shell 命令都具有程序风格,该风格始终将表的名称作
我是一名优秀的程序员,十分优秀!