fs 模块 - 本地文件的读写能力

fsfilesystem的缩写,该模块提供本地文件的读写能力,基本上是 POSIX 文件操作命令的简单包装。但是,这个模块几乎对所有操作提供异步和同步两种操作方式,供开发者选择。

readFile(),readFileSync()

readFile方法用于异步读取数据。

1
2
3
4
fs.readFile("./image.png", function (err, buffer) {
if (err) throw err;
process(buffer);
});

readFile方法的第一个参数是文件的路径,可以是绝对路径,也可以是相对路径。注意,如果是相对路径,是相对于当前进程所在的路径(process.cwd()),而不是相对于当前脚本所在的路径。

readFile方法的第二个参数是读取完成后的回调函数。该函数的第一个参数是发生错误时的错误对象,第二个参数是代表文件内容的Buffer实例。

readFileSync方法用于同步读取文件,返回一个字符串。

1
2
3
4
5
6
var text = fs.readFileSync(fileName, "utf8");

// 将文件按行拆成数组
text.split(/\r?\n/).forEach(function (line) {
// ...
});

readFileSync方法的第一个参数是文件路径,第二个参数可以是一个表示配置的对象,也可以是一个表示文本文件编码的字符串。默认的配置对象是{ encoding: null, flag: 'r' },即文件编码默认为null,读取模式默认为r(只读)。如果第二个参数不指定编码(encoding),readFileSync方法返回一个Buffer实例,否则返回的是一个字符串。

不同系统的行结尾字符不同,可以用下面的方法判断。

1
2
3
4
5
// 方法一,查询现有的行结尾字符
var EOL = fileContents.indexOf("\r\n") >= 0 ? "\r\n" : "\n";

// 方法二,根据当前系统处理
var EOL = process.platform === "win32" ? "\r\n" : "\n";

writeFile(),writeFileSync()

writeFile方法用于异步写入文件。

1
2
3
4
fs.writeFile("message.txt", "Hello Node.js", err => {
if (err) throw err;
console.log("It's saved!");
});

上面代码中,writeFile方法的第一个参数是写入的文件名,第二个参数是写入的字符串,第三个参数是回调函数。

回调函数前面,还可以再加一个参数,表示写入字符串的编码(默认是utf8)。

1
fs.writeFile("message.txt", "Hello Node.js", "utf8", callback);

writeFileSync方法用于同步写入文件。

1
fs.writeFileSync(fileName, str, "utf8");

它的第一个参数是文件路径,第二个参数是写入文件的字符串,第三个参数是文件编码,默认为 utf8。

exists(path, callback)

exists 方法用来判断给定路径是否存在,然后不管结果如何,都会调用回调函数。

1
2
3
fs.exists("/path/to/file", function (exists) {
util.debug(exists ? "it's there" : "no file!");
});

上面代码表明,回调函数的参数是一个表示文件是否存在的布尔值。

需要注意的是,不要在open方法之前调用exists方法,open 方法本身就能检查文件是否存在。

下面的例子是如果给定目录存在,就删除它。

1
2
3
4
if (fs.existsSync(outputFolder)) {
console.log("Removing " + outputFolder);
fs.rmdirSync(outputFolder);
}

mkdir(),writeFile(),readFile()

mkdir 方法用于新建目录。

1
2
3
4
5
var fs = require("fs");

fs.mkdir("./helloDir", 0777, function (err) {
if (err) throw err;
});

mkdir 接受三个参数,第一个是目录名,第二个是权限值,第三个是回调函数。

writeFile 方法用于写入文件。

1
2
3
4
5
6
var fs = require("fs");

fs.writeFile("./helloDir/message.txt", "Hello Node", function (err) {
if (err) throw err;
console.log("文件写入成功");
});

readFile 方法用于读取文件内容。

1
2
3
4
5
6
var fs = require("fs");

fs.readFile("./helloDir/message.txt", "UTF-8", function (err, data) {
if (err) throw err;
console.log(data);
});

上面代码使用 readFile 方法读取文件。readFile 方法的第一个参数是文件名,第二个参数是文件编码,第三个参数是回调函数。可用的文件编码包括“ascii”、“utf8”和“base64”。如果没有指定文件编码,返回的是原始的缓存二进制数据,这时需要调用 buffer 对象的 toString 方法,将其转为字符串。

1
2
3
4
5
var fs = require("fs");
fs.readFile("example_log.txt", function (err, logData) {
if (err) throw err;
var text = logData.toString();
});

readFile 方法是异步操作,所以必须小心,不要同时发起多个 readFile 请求。

1
2
3
4
5
for (var i = 1; i <= 1000; i++) {
fs.readFile("./" + i + ".txt", function () {
// do something with the file
});
}

上面代码会同时发起 1000 个 readFile 异步请求,很快就会耗尽系统资源。

mkdirSync(),writeFileSync(),readFileSync()

这三个方法是建立目录、写入文件、读取文件的同步版本。

1
2
3
4
5
fs.mkdirSync("./helloDirSync", 0777);
fs.writeFileSync("./helloDirSync/message.txt", "Hello Node");
var data = fs.readFileSync("./helloDirSync/message.txt", "UTF-8");
console.log("file created with contents:");
console.log(data);

对于流量较大的服务器,最好还是采用异步操作,因为同步操作时,只有前一个操作结束,才会开始后一个操作,如果某个操作特别耗时(常常发生在读写数据时),会导致整个程序停顿。

readdir(),readdirSync()

readdir方法用于读取目录,返回一个所包含的文件和子目录的数组。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
fs.readdir(process.cwd(), function (err, files) {
if (err) {
console.log(err);
return;
}

var count = files.length;
var results = {};
files.forEach(function (filename) {
fs.readFile(filename, function (data) {
results[filename] = data;
count--;
if (count <= 0) {
// 对所有文件进行处理
}
});
});
});

readdirSync方法是readdir方法的同步版本。下面是同步列出目录内容的代码。

1
2
3
4
5
6
7
8
9
var files = fs.readdirSync(dir);
files.forEach(function (filename) {
var fullname = path.join(dir, filename);
var stats = fs.statSync(fullname);
if (stats.isDirectory()) filename += "/";
process.stdout.write(
filename + "\t" + stats.size + "\t" + stats.mtime + "\n"
);
});

stat()

stat 方法的参数是一个文件或目录,它产生一个对象,该对象包含了该文件或目录的具体信息。我们往往通过该方法,判断正在处理的到底是一个文件,还是一个目录。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
var fs = require("fs");

fs.readdir("/etc/", function (err, files) {
if (err) throw err;

files.forEach(function (file) {
fs.stat("/etc/" + file, function (err, stats) {
if (err) throw err;

if (stats.isFile()) {
console.log("%s is file", file);
} else if (stats.isDirectory()) {
console.log("%s is a directory", file);
}
console.log("stats: %s", JSON.stringify(stats));
});
});
});

watchfile(),unwatchfile()

watchfile 方法监听一个文件,如果该文件发生变化,就会自动触发回调函数。

1
2
3
4
5
6
7
8
9
10
11
12
var fs = require("fs");

fs.watchFile("./testFile.txt", function (curr, prev) {
console.log("the current mtime is: " + curr.mtime);
console.log("the previous mtime was: " + prev.mtime);
});

fs.writeFile("./testFile.txt", "changed", function (err) {
if (err) throw err;

console.log("file write complete");
});

unwatchfile方法用于解除对文件的监听。

createReadStream()

createReadStream方法往往用于打开大型的文本文件,创建一个读取操作的数据流。所谓大型文本文件,指的是文本文件的体积很大,读取操作的缓存装不下,只能分成几次发送,每次发送会触发一个data事件,发送结束会触发end事件。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
var fs = require("fs");

function readLines(input, func) {
var remaining = "";

input.on("data", function (data) {
remaining += data;
var index = remaining.indexOf("\n");
var last = 0;
while (index > -1) {
var line = remaining.substring(last, index);
last = index + 1;
func(line);
index = remaining.indexOf("\n", last);
}

remaining = remaining.substring(last);
});

input.on("end", function () {
if (remaining.length > 0) {
func(remaining);
}
});
}

function func(data) {
console.log("Line: " + data);
}

var input = fs.createReadStream("lines.txt");
readLines(input, func);

createWriteStream()

createWriteStream方法创建一个写入数据流对象,该对象的write方法用于写入数据,end方法用于结束写入操作。

1
2
3
4
5
var out = fs.createWriteStream(fileName, {
encoding: "utf8"
});
out.write(str);
out.end();

createWriteStream方法和createReadStream方法配合,可以实现拷贝大型文件。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
function fileCopy(filename1, filename2, done) {
var input = fs.createReadStream(filename1);
var output = fs.createWriteStream(filename2);

input.on("data", function (d) {
output.write(d);
});
input.on("error", function (err) {
throw err;
});
input.on("end", function () {
output.end();
if (done) done();
});
}