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.lua 文件的第一行信息,并在该文件最后一行添加了 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.lua 文件的第一行信息,并在该文件最后一行添加了 lua 的注释。如我这边输出的是:
-- test.lua 文件
read 的参数与简单模式一致。
其他方法:
file:seek(optional whence, optional offset): 设置和获取当前文件位置,成功则返回最终的文件位置(按字节),失败则返回nil加错误信息。参数 whence 值可以是:
- "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
游戏萌新
986***[email protected]
使用 *n 作为参数读取文件中的数字的时候,只有文件中第一个字符是数字(或者空格加数字)的情况下才能读取到并返回这个数字:
test.txt 文件内容:
输出:
游戏萌新
986***[email protected]
王风
185***[email protected]
1.若使用LuaStudio,其中io.open()的默认查找路径是LuaStudio软件的根目录下,若需要open其他路径的文件,使用绝对路径即可;
2.需要注意的是,简单模式下,io.read()不能放在io.write()后面,否则,虽然不会报错,但会打印出奇怪的字符,打开文件看,会发现多了一串乱码。
3.若需要io.write后io.read,则需要先关闭文件,然后再打开文件查看
4.简单模式下,当使用io.close()关闭文件的时候,之前设置默认输入文件和默认输出文件会失效,当你使用io.open()重新打开文件时,则需要重新设置
5.关于 *n ,还有一点需要注意,若数字中有非数字的字符的话,会返回该字符前的数字,如:
t.lua文件内容是:
输出:
王风
185***[email protected]
伟哥
wah***[email protected]
感觉二楼说法有误: io.read() 不能放在 io.write() 后面,否则,虽然不会报错,但会打印出奇怪的字符,打开文件看,会发现多了一串乱码。
我的分析如下:
1. 这是因为执行 io.write 后,写入的数据只是在缓冲区中,并未真正写入到文件中,当执行 io.flush 后,缓冲区的数据才会真正写入文件中,或者执行 io.close 方法时,在关闭文件前,也会先将缓冲区数据写入文件。
2. 当执行 io.open 方法时,会有一个类似 C/C++ 的指针的文件位置指向,会将当前要读取或写入的数据写入到指向的位置,然后指向的位置进行偏移。例如:
3.所以才会有一个 file:seek() 的方法,可以让你修改当前文件流操作的指向位置。希望大家能理解文件流操作的位置概念(其实可以理解为指针)
伟哥
wah***[email protected]
围观群众
128***[email protected]
文件操作中的绝对地址,从资源管理器复制的话,需要反斜杠前面再加一个,因为反斜杠本身是转义字符,下面的文件路径:
应该是:
围观群众
128***[email protected]
PMonsterR
602***[email protected]
验证了一下三楼的说法,以代码的情况来看,不用调用 io.flush 也能读取到写入的数据,但是不确定的是 file:seek 或者是 io.read 会不会本身就有 io.flush 类似的效果。
在二楼的代码基础上做了以下修改:
PMonsterR
602***[email protected]