Lua 最佳实践:使用 popen() 关闭变量

我是 Lua 的新手,对于 Lua 的内存管理有些疑问。

问题一:当使用 io.popen() 函数时,我发现许多 Lua 程序员在使用 popen() 函数后编写了一个 close 语句。我想知道原因是什么?例如,看一下这段代码:

handle = io.popen("ls -a")
output = handle:read("*all")
handle:close()
print(output)

handle = io.popen("date")
output = handle:read("*all")
handle:close()
print(output)

我听说 Lua 可以自己管理内存。那么,我真的需要像上面那样编写 handle:close 吗?如果我忽略了 handle:close() 语句并像以下方式编写它,那么内存会发生什么?

handle = io.popen("ls -a")
handle = io.popen("date")
output = handle:read("*all")

问题二:从 问题一 的代码中,在内存使用方面,我们可以在末尾只写一行 handle:close() 语句而不是两行吗?:

handle = io.popen("ls -a")
output = handle:read("*all")
-- handle:close() -- 暂不要关闭,在最后关闭
print(output)
handle = io.popen("date") -- 这里使用先前将变量 `handle` 赋值
output = handle:read("*all")
handle:close()  -- 只需一条语句即可关闭以上所有
print(output)

你可以看到,我没有在第一条 io.popen 语句中关闭它,但我在最后关闭它,这会使程序变慢吗?

点赞
用户7396148
用户7396148

Lua会在垃圾回收器回收它时自动关闭文件句柄。

Lua手册5.4:file:close

关闭文件。请注意,当它们的句柄被垃圾回收后,文件会自动关闭,但需要不可预知的时间才能发生。

但是,最好的做法是当你处理好句柄后立即关闭它,因为垃圾回收器需要不确定的时间才能关闭句柄。

这并不是关于内存的问题,而是关于打开文件句柄的资源更为有限,例如在windows计算机上大约只有512个句柄池,供所有运行在其中的应用程序使用。


至于第二个问题,当你重新分配一个变量并且没有其他保留对上一个值的引用时,该值将最终被垃圾回收器回收。

2020-11-04 14:26:13
用户4984564
用户4984564

问题 1

在这种情况下,close 不是为了内存原因,而是为了关闭文件。当文件句柄被收集时,它将自动关闭,但如果程序不生成太多垃圾(一些程序员专门进行优化),GC 可能在程序处理文件后一段时间内不会运行,文件会保持打开状态。

另外,如果变量仍在作用域中,则 GC 将无法在作用域结束之前收集它,这可能需要很长时间。

问题 2

那样做是不行的。方法在值上调用而不是变量,因此当您将一个新值分配给一个变量时,旧值就会消失。对新值调用方法不会影响以前存储在变量中的任何其他值。

2020-11-04 15:30:16