如何在跳出循环时清理通用迭代器函数中的内容
2017-2-27 11:6:45
收藏:0
阅读:88
评论:1
我正在C语言中实现一个通用的迭代器函数。返回通用循环的迭代器函数的函数首先使用 lua_newuserdata() 来分配迭代器函数的状态信息,如下所示:
struct mystate *s = (struct mystate *) lua_newuserdata(L, sizeof(struct mystate));
s->firstcall = 1;
然后,我将该指针作为闭包的upvalue推入C的栈中,如下所示:
lua_pushvalue(L, lua_gettop(L));
lua_pushcclosure(L, iteratorfunction, 1);
然后,我的迭代器函数从第一个upvalue中检索指针,并对其进行一些分配,如下所示:
static int iteratorfunction(lua_State *L)
{
struct mystate *s = (struct mystate *) lua_touserdata(L, lua_upvalueindex(1));
if(s->firstcall) {
s->file = fopen(...);
s->data = malloc(...);
...
s->firstcall = 0;
}
...
}
现在我的问题是:如果脚本在完成之前使用“break”退出通用for循环,如何确保在s->file 和s->data上正确释放内存?在这种情况下,我的iteratorfunction不能执行清理,因为它没有完全被调用时,通用for循环就已经退出了。
具体来说,如果脚本在完成之前使用“break”退出我的通用for循环,怎样才能确保在s->file 上调用fclose(),在s->data 上调用free()?
我查看了Lua的源代码和io.lines似乎使用元表和垃圾回收器确保文件句柄被关闭,但我不太理解这是如何工作的。看起来相当复杂,我不确定我是否应该以类似的方式进行操作,还是是否有更简单的解决方案适用于我的情况。
请注意,我仍在使用Lua 5.0,因此任何解决方案的建议都应牢记这一点。谢谢!
点赞
评论区的留言会收到邮件通知哦~
推荐文章
- Lua 虚拟机加密load(string.dump(function)) 后执行失败问题如何解决
- 我想创建一个 Nginx 规则,禁止访问
- 如何将两个不同的lua文件合成一个 东西有点长 大佬请耐心看完 我是小白研究几天了都没搞定
- 如何在roblox studio中1:1导入真实世界的地形?
- 求解,lua_resume的第二次调用继续执行协程问题。
- 【上海普陀区】内向猫网络招募【Skynet游戏框架Lua后端程序员】
- SF爱好求教:如何用lua实现游戏内调用数据库函数实现账号密码注册?
- Lua实现网站后台开发
- LUA错误显式返回,社区常见的规约是怎么样的
- lua5.3下载库失败
- 请问如何实现文本框内容和某个网页搜索框内容连接,并把网页输出来的结果反馈到另外一个文本框上
- lua lanes多线程使用
- 一个kv数据库
- openresty 有没有比较轻量的 docker 镜像
- 想问一下,有大佬用过luacurl吗
- 在Lua执行过程中使用Load函数出现问题
- 为什么 neovim 里没有显示一些特殊字符?
- Lua比较两个表的值(不考虑键的顺序)
- 有个lua简单的项目,外包,有意者加微信 liuheng600456详谈,最好在成都
- 如何在 Visual Studio 2022 中运行 Lua 代码?

回答我的问题,我现在使用了终结器(
__gc元方法)建议使用Egor。在代码中它看起来像这样:首先我们需要创建一个元表,其中我们可以使用
__gc来清理:#define PRIVATEHANDLE "PRIVATE*" luaL_newmetatable(L, PRIVATEHANDLE); lua_pushliteral(L, "__index"); lua_pushvalue(L, -2); lua_rawset(L, -3); lua_pushstring(L, "__gc"); lua_pushcclosure(L, iteratorfunction_gc, 0); lua_settable(L, -3); lua_pop(h, 1);然后我们需要将用户数据与元表相关联,以便Lua决定删除我们的用户数据时调用我们的
__gc方法,因此我们执行:struct mystate *s = (struct mystate *) lua_newuserdata(L, sizeof(struct mystate)); memset(s, 0, sizeof(struct mystate)); luaL_getmetatable(L, PRIVATEHANDLE); lua_setmetatable(L, -2);最后,我们需要实现
iteratorfunction_gc来进行实际清理。这可能看起来像这样:static int iteratorfunction_gc(lua_State *L) { struct mystate *s = (struct mystate *) luaL_checkudata(L, 1, FILEHANDLE); if(s->file) fclose(s->file); if(s->data) free(s->data); ...这里可以补充其他的清理操作... return 0; }经过测试,这能够非常好地工作。问题解决了。不知道为什么有人想关闭这个问题。