fs
是filesystem
的缩写,该模块提供本地文件的读写能力,基本上是 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 ( ) { }); }
上面代码会同时发起 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(); }); }