Lua - io.open()最多只有2 GB?

时间:2014-11-05 10:21:41

标签: windows file-io lua

我使用Lua脚本来确定文件大小:

local filesize=0
local filePath = "somepath.bin"
local file,msg = io.open(filePath, "r")
if file then
    filesize=file:seek("end")
    file:close()
    filePresent = true
end

但是,这似乎仅适用于最大2GB的文件。对于较大的文件,filesize始终为nilio.open有任何限制吗?如果是这样,我怎么能解决这个问题?

在Windows Server 2008 R2 64位上运行Lua 5.1.4

3 个答案:

答案 0 :(得分:5)

问题不在io.open,而在file:seek。您可以像这样检查错误:

filesize, err = file:seek("end")
if not filesize then
    print(err)
end

错误消息可能是Invalid argument。这是因为对于大于2GB的文件,其大小超过了32位long可以容纳的大小,这导致C函数fseek无法正常工作。

在POSIX系统中,Lua使用fseeko,其大小为off_t,而不是long中的fseek。在Windows中,有一个_fseeki64,我猜这类似的工作。如果这些不可用,则使用fseek,这会导致问题。


相关来源是liolib.c(Lua 5.2)。正如@lhf指出的那样,在Lua 5.1中,总是使用fseeksource)。升级到Lua 5.2可能会解决问题。

答案 1 :(得分:3)

在内部,Lua使用ISO C函数long int ftell(FILE *stream);来确定file:seek()的返回值。 Windows上long int总是32位,所以你在这里运气不好。如果可以,您应该使用一些外部库来检测文件大小 - 我建议luafilesystem

答案 2 :(得分:0)

在旧的Lua版本上(file:seek()限制为2Gb),您可以要求cmd.exe获取文件大小:

function filesize(filename)
   -- returns file size (or nil if the file doesn't exist or unable to open)
   local command = 'cmd /d/c for %f in ("'..filename..'") do @echo(%~zf'
   return tonumber(io.popen(command):read'*a')
end

print(filesize[[C:\Program Files\Windows Media Player\wmplayer.exe]])
         --> 73728
print(filesize[[E:\City.of.the.Living.Dead.1980.720p.BluRay.x264.Skazhutin.mkv]])
         --> 8505168882