理解lua中Weak Table

如果你对lus语言的中weak table不明白的话,那这篇文章应该对你有帮助。

所有脚本语言几乎都有垃圾回收器(GC),当然lua也有。

不明白GC不要紧,下面让我来解释:
这一切要先从现代编程语言的鼻祖 — C语言开始。

如果你在C中要使用变量,可以这样定义变量,如:
----------------------------------------------------
void hello(){
int a = 0; // 局部变量声明
}
----------------------------------------------------
局部变量a的生与死是已经确定的,就是{括号和}括号之间。
这样会很安全,因为函数在执行完后会清栈(清理变量),所以你不用去考虑变量的生与死,但,它不够灵活,因为它的作用域被界限了,它只能在这个函数里被使用。

换种手段,定义全局变量,如下:
----------------------------------------------------
static int a = 0;// 全局变量声明

void hello1(){
a = 1;
}
void hello2(){
a = 2;
}
----------------------------------------------------
很显然,这样很灵活,到哪里都可以使用。
不过自从它生之后,就不会结束了,直到程序关闭时它才结束。
这意味着,程序运行期间无论如何它都将存在。
你面对的将是一个永恒的变量,如果你定义了10,100,1000个...那将会是一堆...(僵尸?)

这样都很不好,你无法控制变量的生与死,无法达到灵活处理变量生与死的目的。
想亲手控制变量的生与死?C便提供了一个函数实现你的梦想 - malloc,如下:
----------------------------------------------------
int* p = (int *)malloc(sizeof(int)); // 手动创建 — 生

这样p便存储了申请过来的这块内存首地址,现在就可以使用了,如下:

*p = 1;

当你不再使用这块内存的时候,那你就使用free函数释放(清理)它,否则它将一直存在。

free(p);   // 手动释放 — 死
----------------------------------------------------
如果你不这样做,那内存将会增长直到程序崩溃(世界就乱了)。

这样子看,第3个方法很好很灵活,但是一旦你定义了这种变量多了,你将成为上帝 — 你必须管理每个变量的生与死,哪怕是1,10,1000。一个都不能漏,否则世界的平衡就乱了(有人不死啦),换句话你程序也将挂了。

这样看来很显然不是每个人都喜欢或者能做上帝,在这样的情况下,有一个东西横空出世了 — GC!
它就是这个问题的救世主,下面详细介绍:

GC - Garbage Collectoion - 垃圾回收器

它的作用就是变量你随意定义,手尾我来跟,你只须定义变量的生,变量的死由我来负责。
那你不经地问:那你怎么知道这个变量我不用了?

那这个问题就比较复杂,目前GC的实现有好多种,敝人只知道引数GC,下面就用lua来说说它。
先说明一点:在脚本语言中变量的声明都是使用上面说的第三种方法 - malloc来手动创建内存块。

lua代码:
----------------------------------------------------
function test()
local a = {x=1};   -- GC将给"{x=1}"手动创建的这块内存块增加一个引用 - b,引用数为1
end

test(); -- 函数执行完后清栈,局部变量a被清掉,GC将给"{x=1}"这块内存将减少一个引用,引用数为0,引用数为0时GC就清理掉"{x=1}"这块内存。

print(a);   -- nil
--------------------------------------
再看下面:
----------------------------------------------------
b = 0;   -- 外部变量
function test()
local a = {x=1};
b = a;   -- GC将给"{x=1}"这块内存再加一个引用 - a,引用数为2
end
test(); -- 函数执行完后清栈,"{x=1}"这块内存引用数减1还剩1,那这块内存块不用清理。

print(b); -- table: 003BAAD8
----------------------------------------------------

上面的例子你应该大致清楚GC的工作原理了吧,下面看最后此文的关键:lua - weak table

在lua中任何对内存块的引用都会使引用数加1,但有一个例外:weak table,它对内存块的引用不会使引用增1
看下面代码:
----------------------------------------------------
a = {};
b = {};    
setmetatable(a,b); -- 设置a为weak table
b.__mode = 'k';

key = {};    -- 增加"{}"内存块的一个引用 - key,引用数1
a[key] = 1; -- weak table引用不增引数,所以"{}"内存块的引数还为1
key = {}    -- 改变key指向新增的"{}"内存块,上面的"{}"内存块引数减一为0
a[key] = 2     -- 如上上一样

collectgarbage();   -- 调用GC,清掉weak表中没有引用的内存

for k,v in pairs(a) do
print(v);    -- 只输出一个2
end
----------------------------------------------------
如果a不是weak table而是普通的table,那么a将会对"{}"内存块的引数加1,
去掉"b.__mode = 'k';"这句,你将会看到输出1和2。
看最后一个例子:
----------------------------------------------------
a = {}
b = {}
setmetatable(a,b)
b.__mode = "k"
function test()
    local key1 = {};
    a[key1] = 1;
    local key2 = {};
    a[key2] = 2;
end
test(); -- 函数执行完后key1和key2它们指向的内存块引数都减一为0

collectgarbage() -- 故全部都被清空
for k,v in pairs(a) do
print(v); -- 没有任何输出
end
----------------------------------------------------
以上是我自己对gc的理解(没去查资料^_^),仅供参考...

后语:GC虽然是救世主,但不是万能的救世主,在回收变量中存在一些问题,如效率、循环引用等等需要去关注...

 

本文转自:http://hi.baidu.com/xuyuqiang/blog/item/29aa10f2529cba14b17ec51b.html

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐


1.github代码实践源代码是lua脚本语言,下载th之后运行thmain.lua-netTypevgg-savevgg_cifar10/-S0.0001,报错: 试看看安装lua:报错了,参考这篇文章:ubuntu18.04安装lua的步骤以及出现的问题_weixin_41355132的博客-CSDN博客问题解决,安装成功:情况并没有好转,出现相
此文为搬运帖,原帖地址https://www.cnblogs.com/zwywilliam/p/5999924.html前言在看了uwa之前发布的《Unity项目常见Lua解决方案性能比较》,决定动手写一篇关于lua+unity方案的性能优化文。整合lua是目前最强大的unity热更新方案,毕竟这是唯一可以支持ios热更新的办法。然而作
Rime输入法通过定义lua文件,可以实现获取当前时间日期的功能。1.TIMERime是一款可以高度自定义的输入法,相关教程可以查看往期文章,关于时间获取是指输入一个指定关键字,输出当前时间,效果如下(我定义了time关键字):实现如下:①在用户文件夹中新建一个rime.lua文件加入如下代码 ti
localfunctiongenerate_action(params)localscale_action=cc.ScaleTo:create(params.time,params.scale_x,params.scale_y)localfade_action=cc.FadeIn:create(params.time)returncc.Spawn:create(scale_action,fade_action)end
2022年1月11日13:57:45 官方:https://opm.openresty.org/官方文档:https://opm.openresty.org/docs#table-of-contents为什么建议使用opm不建议使用luarocks?http://openresty.org/cn/using-luarocks.html官方解释:请注意!LuaRocks并不是OpenResty官方推荐的装包方式。LuaRoc
在Lua中的table(表),就像c#中的HashMap(哈希表),key和value一一对应。元表:table的一个操作的拓展,里面包含关联了对应的方法,元方法就是其中一个。元方法:当你通过键来访问table的时候,如果这个键没有值,那么Lua就会寻找该table的metatable(假定有metatable)中的__index键。如果__inde
表排序:table.sort(list[,comp])参数list:指定表,可选参数comp:排序函数,无参数时通常按升序排序。排序函数针对表中连续的序列,其间不可以存在空洞或nil,排序函数需要两个形参(对应表中每次参加比较的两个数据),需要一个比较两个形参表达式的返回值,不能含有等于关系,例如>=,<=,==。do
一、安装lua环境1.1安装依赖包[root@centos7~]#yuminstallgccreadline-devel1.2下线lua源码包并解压[root@centos7~]#wgethttp://www.lua.org/ftp/lua-5.3.5.tar.gz[root@centos7~]#tarxvflua-5.3.5.tar.gz-C/usr/local/src1.3进行编译[root@centos7~]
官网OpenResty® 是一个基于 Nginx 与Lua的高性能Web平台,其内部集成了大量精良的Lua库、第三方模块以及大多数的依赖项。用于方便地搭建能够处理超高并发、扩展性极高的动态Web应用、Web服务和动态网关。OpenResty® 通过汇聚各种设计精良的 Nginx 模块(主要由
表参考《lua程序设计》可以认为,表是一种动态分配的对象,程序只能操作指向表的引用(或指针)。除此以外,Lua语言不会进行隐藏的拷贝(hiddencopies)或创建新的表--创建表a={}--创建空表k="x"a[k]=10--键“x”值10a[20]="great"--键20值“great”print(a["x"])-->10
https://github.com/galenho/crossover.git一个跨平台的lua游戏服务器开发框架,该框架采用多线程并发来处理消息,开发者只需要调用相应的接口函数并绑定相应的回调函数即可,在逻辑层表现为单线程的开发模式,使开发者易用,易调试,易维护,易扩展,同时拥有快速的响应能力。   框架使用面
参考链接:https://www.runoob.com/lua/lua-metatables.htmlhttps://www.jianshu.com/p/cb945e7073a3 元表是一个table,可以让我们改变table的行为,每个行为有对应的元方法例如,对table进行设置键值,查找键值,运算等,就会触发对应的元方法1--__index:table被访问时,如果找不到这
https://github.com/yuin/gopher-luahttps://github.com/yuin/gopher-lua Lua5.1ReferenceManual-contentshttp://www.lua.org/manual/5.1/ go中使用luapackagemainimport( lua"github.com/yuin/gopher-lua")funcmain(){ l:=lua.NewState() d
编译问题不要留到运行时才跑出来啊。早上9:00-中午3:00,6个小时,服了自己了。 写了一个测试,springboot+redis+lua执行到redisTemplate.execute(redisScript,idList)的时候一直报错,integer无法转换为string。我一直以为是lua脚本写错了,翻文档翻过来又翻过去,写法变了又变,还是解
        。。是字符串连接符,字典用=号连接,  注意fordoend都是连一起,  注意ifthen,  如果local在函数里,是可以访问,非local出了函数一样能用,  doend代码块也是一样,    注意点号表示,只能key是字符串,  注意括号不是必须
C语言与Lua之间的相互调用详解写一个C调用Lua的Demo编译运行C语言调用Lua编译问题总结正确的编译命令问题1:缺少-lm参数问题2:缺少-ldl参数​1、为什么会出现undefinedreferenceto‘xxxxx’错误?​2、-l参数和-L参数写一个C调用Lua的Demo编译运行add.c内容//你需要
1、动态输出打开E:\study\openresty\openresty-1.19.9.1-win64目录下的confginx.conf文件在server中增加一下代码 location/hello{ default_typetext/html; content_by_lua'ngx.say("<p>hello,world</p>")'; }运行后,效果如下图localhost
参见:lipp/lua-websockets:WebsocketsforLua.(github.com)github网址可能需手动转换lipp.github.com/lua-websockets/>github.com/lipp/lua-websocketswebsockets为底层的类似于TCP、UDP的socket(实现上基于更底层的socket),不同于上层的webserver服务端(Service)需并行地支持多
lua发送消息到rabbitmq,我们选择类库lua-resty-rabbitmqstomp 来完成这个任务。类库安装:进入nginx.conf中 lua_package_path 中对应的目录下的resty目录(没有则创建),执行:wget-chttps:/aw.githubusercontent.com/wingify/lua-resty-rabbitmqstomp/master/libes
1Lua介绍Lua是一门以其性能著称的脚本语言,被广泛应用在很多方面。Lua一般用于嵌入式应用,现在越来越多应用于游戏当中,魔兽世界,愤怒的小鸟都有用到。优势Lua极易嵌入到其他程序,可当做一种配置语言。提升应用性能,比如:游戏脚本,nginx,wireshark的脚本兼容性强,可以直接使用C