Lua 文件 I/O 操作

2019-05-04 23:57|来源: 网路

Lua 文件 I/O

Lua I/O 库用于读取和处理文件。分为简单模式(和C一样)、完全模式。

  • 简单模式(simple model)拥有一个当前输入文件和一个当前输出文件,并且提供针对这些文件相关的操作。

  • 完全模式(complete model) 使用外部的文件句柄来实现。它以一种面对对象的形式,将所有的文件操作定义为文件句柄的方法

简单模式在做一些简单的文件操作时较为合适。但是在进行一些高级的文件操作的时候,简单模式就显得力不从心。例如同时读取多个文件这样的操作,使用完全模式则较为合适。

打开文件操作语句如下:

file = io.open (filename [, mode])

mode 的值有:

模式 描述
r 以只读方式打开文件,该文件必须存在。
w 打开只写文件,若文件存在则文件长度清为0,即该文件内容会消失。若文件不存在则建立该文件。
a 以附加的方式打开只写文件。若文件不存在,则会建立该文件,如果文件存在,写入的数据会被加到文件尾,即文件原先的内容会被保留。(EOF符保留)
r+ 以可读写方式打开文件,该文件必须存在。
w+ 打开可读写文件,若文件存在则文件长度清为零,即该文件内容会消失。若文件不存在则建立该文件。
a+ 与a类似,但此文件可读可写
b 二进制模式,如果文件是二进制文件,可以加上b
+ 号表示对文件既可以读也可以写

简单模式

简单模式使用标准的 I/O 或使用一个当前输入文件和一个当前输出文件。

以下为 file.lua 文件代码,操作的文件为test.lua(如果没有你需要创建该文件),代码如下:

-- 以只读方式打开文件
file = io.open("test.lua", "r")

-- 设置默认输入文件为 test.lua
io.input(file)

-- 输出文件第一行
print(io.read())

-- 关闭打开的文件
io.close(file)

-- 以附加的方式打开只写文件
file = io.open("test.lua", "a")

-- 设置默认输出文件为 test.lua
io.output(file)

-- 在文件最后一行添加 Lua 注释
io.write("--  test.lua 文件末尾注释")

-- 关闭打开的文件
io.close(file)

执行以上代码,你会发现,输出了 test.ua 文件的第一行信息,并在该文件最后一行添加了 lua 的注释。如我这边输出的是:

-- test.lua 文件

在以上实例中我们使用了 io."x" 方法,其中 io.read() 中我们没有带参数,参数可以是下表中的一个:

模式 描述
"*n" 读取一个数字并返回它。例:file.read("*n")
"*a" 从当前位置读取整个文件。例:file.read("*a")
"*l"(默认) 读取下一行,在文件尾 (EOF) 处返回 nil。例:file.read("*l")
number 返回一个指定字符个数的字符串,或在 EOF 时返回 nil。例:file.read(5)

其他的 io 方法有:

  • io.tmpfile():返回一个临时文件句柄,该文件以更新模式打开,程序结束时自动删除

  • io.type(file): 检测obj是否一个可用的文件句柄

  • io.flush(): 向文件写入缓冲中的所有数据

  • io.lines(optional file name): 返回一个迭代函数,每次调用将获得文件中的一行内容,当到文件尾时,将返回nil,但不关闭文件


完全模式

通常我们需要在同一时间处理多个文件。我们需要使用 file:function_name 来代替 io.function_name 方法。以下实例演示了如何同时处理同一个文件:

-- 以只读方式打开文件
file = io.open("test.lua", "r")

-- 输出文件第一行
print(file:read())

-- 关闭打开的文件
file:close()

-- 以附加的方式打开只写文件
file = io.open("test.lua", "a")

-- 在文件最后一行添加 Lua 注释
file:write("--test")

-- 关闭打开的文件
file:close()

执行以上代码,你会发现,输出了 test.ua 文件的第一行信息,并在该文件最后一行添加了 lua 的注释。如我这边输出的是:

-- test.lua 文件

read 的参数与简单模式一致。

其他方法:

  • file:seek(optional whence, optional offset): 设置和获取当前文件位置,成功则返回最终的文件位置(按字节),失败则返回nil加错误信息。参数 whence 值可以是:

    不带参数file:seek()则返回当前位置,file:seek("set")则定位到文件头,file:seek("end")则定位到文件尾并返回文件大小

    • "set": 从文件头开始

    • "cur": 从当前位置开始[默认]

    • "end": 从文件尾开始

    • offset:默认为0

  • file:flush(): 向文件写入缓冲中的所有数据

  • io.lines(optional file name): 打开指定的文件filename为读模式并返回一个迭代函数,每次调用将获得文件中的一行内容,当到文件尾时,将返回nil,并自动关闭文件。
    若不带参数时io.lines() <=> io.input():lines(); 读取默认输入设备的内容,但结束时不关闭文件,如

    for line in io.lines("main.lua") do
    
      print(line)
    
      end

以下实例使用了 seek 方法,定位到文件倒数第 25 个位置并使用 read 方法的 *a 参数,即从当期位置(倒数第 25 个位置)读取整个文件。

-- 以只读方式打开文件
file = io.open("test.lua", "r")

file:seek("end",-25)
print(file:read("*a"))

-- 关闭打开的文件
file:close()

我这边输出的结果是:

st.lua 文件末尾--test

相关问答

更多
  • 很难理解你到底是要什么,是想用eclipse写lua代码,还是用打开已有的lua代码。要是写就和建立其他工程一样的了。
  • 尝试这个: local str = "https://thisisarandomsite.com/some_dir/src/blah/blah/7fd34a0945b036685bbd6cc2583a5c30.jpg" local name = str:match( "([^/]+)$" ) 您可以从本指南中自定义匹配模式。 Try this: local str = "https://thisisarandomsite.com/some_dir/src/blah/blah/7fd34a0945b0366 ...
  • “Lua编程”第8页 在脚本的主体中,表达式“...”导致脚本的参数。 所以a和ns应该接收WOW传入的前两个参数。 Page 8 of "Programming in Lua" In the main body of a script, the expression "..." results in the arguments to the script. So a and ns should receive the first two arguments passed in by WOW.
  • Lua 5.3将 lua_pop 定义为 #define lua_pop(L,n) lua_settop(L, -(n)-1) 和lua_settop可以在这里看到; 它将用nil填充堆栈槽,因此值可以是gc'd。 所以它是带有arg 1的O(1)。 您可以读取O(1)中的每个值,并且N值的弹出值是O(N)。 所以,你的两种方法应该大致相同。 Re:读取O(1)中的值,函数index2addr将堆栈偏移量转换为地址。 Lua 5.3 defines lua_pop as #defin ...
  • 由于错误消息告诉您cjson和decode是无法索引或调用的nil值。 require将加载一些文件并运行包含的代码并传递返回值。 如果你运行一个Lua脚本,它的行为就像一个默认返回nil的函数。 因此,除非您指定脚本返回require内容,否则将返回nil 。 我不知道你需要的cjson文件里面有什么,但它显然不会返回想要的json实现但是nil 。 所以cjson中的代码应该返回一个Lua表,其中一个函数存储在键“decode”下。 As the error message tells you cjso ...
  • 现在为了执行lua文件,我运行命令 这通常不是你如何执行Lua文件。 通常,如果你有一些Lua脚本,你可以用下面的命令执行它: lua a.lua 。 你不要输入lua ,然后使用那里的接口来执行它。 使用适当的命令行来执行脚本,可以将字符串参数传递给文件: lua a.lua someParam "Param with spaces" 。 然后, a.lua脚本可以使用标准的Lua ...机制来获取这些参数: local params = {...} params[1] -- first paramete ...
  • 我不知道你配置的细节,但试试这个 require ("src.socket") 你应该从lib的根路径需要一个模块 I don't know the details of you configuration, but try this require ("src.socket") you should require a module from the root path of the lib
  • 如果需要从LuaJava加载/编译字符串,可以使用函数LuaState.LloadString(String source) 。 如果您不想多次从源加载“模块”,则必须为其分配名称并在表中保存一些标志。 您甚至可以提供“卸载”,以便您可以再次从源加载模块。 它可以在Lua中重新实现如下: do local loadedModules = {} -- local so it won't leak to _G function loadModule(name, source) if loaded ...
  • 要将Lua文件打包到Windows上的可执行文件中,您有几个选项。 有srlua ,有来自wxLua的wxLuaFreeze(可用作Windows的二进制文件 ),在这个SO答案中还有更多选项。 本质上,主要的两个选项是:(1)将您的Lua代码附加到预编译的exe文件,以便在运行该exe文件时加载和执行,以及(2)通过编译将您的Lua代码转换为实际可执行文件到字节码,然后到C,然后到你的目标平台。 至于你的MacOS问题, mkdir -p意味着要求mkdir创建中间目录(例如,你要求创建/a/b/c ,如 ...
  • 正如Lua要求的man页面所描述的 ,它在路径中搜索文件。 这个路径可以在C中定义。看看这篇文章: “从C ++ / C设置全局LUA_PATH变量” require函数非常实用,可以加载.lua文件中定义的模块和库。 As decribed by the Lua require man page, it searches for the file in a path. This path can be defined in C. Have a look to this post : "Setting th ...