如果一个table1,里面还有一个table2对象,那么即使将这个table2赋值为nil,它依然不会被垃圾回收因为table1还存有着对这个对象的引用,显然这样的引用是没有必要的。
为了告诉GC,这一部分的引用是没必要的,所以需要将其标记为弱引用。以便GC可以对其进行回收。(小老弟,我是个垃圾)
t1, t2 = {}, {}
arr = {}
arr[1] = t1
arr[2] = t2
t1 = nil --此时我们将t1赋值为nil,表明这一部分的内容我们不需要再使用,可以被回收
collectgarbage() -- 手动执行gc回收操作
for k, v in pairs(arr) do
print(k, v)
end
结果如下所示
1 table: 0053B068
2 table: 0053B158
两个表依然存在,并没有被回收,因为此时表arr还存在对他们的引用
我们使用 __mode 来标明弱引用关系
__mode 有两个参数,“k"和"v” 他们也可以连起来一起使用"kv"
操作符 | 说明 |
---|---|
k | 表的key为弱引用 |
v | 表的value为弱引用 |
以上一个例子,可以重新写成
t1, t2 = {}, {}
arr = {}
arr[1] = t1
arr[2] = t2
setmetatable(arr, {__mode = "v"})
t1 = nil --此时我们将t1赋值为nil,表明这一部分的内容我们不需要再使用,可以被回收
collectgarbage() -- 手动执行gc回收操作
for k, v in pairs(arr) do
print(k, v)
end
结果为
2 table: 007CB158
这里,第一个表被回收了。
以此类推,如果将表作为键,那么选择"k"的模式,则会标记键为弱引用。
而如果标记"kv",则这两个条件同时生效,只要其中一个可被回收则会被直接回收。