fs api(阮老师的api文档) fs是filesystem的缩写,该模块提供本地文件的读写能力,基本上是POSIX文件操作命令的简单包装。但是,这个模块几乎对所有操作提供异步和同步两种操作方式,供开发者选择。
fs模块是唯一一个同时提供同步和异步API的模块。
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 6 7 // 方法一,查询现有的行结尾字符 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 10 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 19 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 6 7 8 9 10 11 12 13 14 15 16 17 18 var out = fs.createWriteStream(fileName, { encoding: 'utf8' }); out.write(str); out.end(); createWriteStream方法和createReadStream方法配合,可以实现拷贝大型文件。 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(); }); }
同步还是异步
要在单线种中创建能够处理高并发的高效程序,采用异步,事件驱动的程序是不错的选择。
理解Stream node.js console 会输出内容到控制台。事实上,console.log内部做了如下处理: 在指定的字符串后面加上\n(换行)字符,并将其写入stdout流中。
process 全局对象中包含了三个流对象。
stdin 标准输入
stdout 标准输出
stderror 标准错误
需求场景描述 fs模块允许我们通过Stream API来对数据进行读写操作。与readFile 及 writeFile方法不同,它对内存的分配不是一次完成的。
有一个大文件,文件内容由上百万行逗号分割文本组成。要完整的读取该文件进行解析。意味着要一次性分配很大的内存。更好的方式应该是一次只读取一块内容,以行尾结束符(“\n”)来切分,然后再逐块进行解析。
Node Stream 就是对上述解决方案的完美实现。
Stream fs.createREadStream方法允许为一个文件创建一个可读的Stream对象。方便理解,举例如下:
1 2 3 fs.readFile('xxx.txt', function(err, contents) { //处理整个文件 });
上面的例子,回调函数必须要等到整个文件读取完毕,载入到RAM,且可用的情况下才会触发。
1 2 3 4 5 6 7 8 9 var stream = fs.createReadStream('xxx.txt'); stream.on('data', function(chunk) { //处理文件的部分内容 }); stream.on('end', function() { //文件读取完毕。 })
应用场景
大文件上传
日志记录(操作系统的打开,关闭文件较低效)
Stream 继承自EventEmitter demo fs初版demo 初级需求
程序在命令行运行。通过终端提供交互给用户输入,输出。
程序启动后,需要显示当前目录下的文件列表。
选择某个文件时,程序需要显示该文件的内容。
选择一个目录时,程序需要显示该目录下的信息。
运行结束后退出程序。
code 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 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 var fs = require('fs'), colors = require('colors'), stdin = process.stdin, stdout = process.stdout, path = require('path'), stats = [], files = fs.readdirSync(process.cwd()); function file(i) { var filename = files[i]; fs.stat( path.resolve(__dirname, filename), function (err, stat) { stats[i] = stat; if (stat.isDirectory()) { console.log(` ${i} ${filename}/`.blue); } else { console.log(` ${i} ${filename}`.green); } if (++i == files.length) { read(); } else { file(i); } } ); } function read() { console.log(``); stdout.write(` Enter your choice: `.yellow); stdin.resume(); stdin.setEncoding('utf8'); stdin.on('data', option); } function option(data) { var filename = files[Number(data)]; if (!filename) { stdout.write(` Enter your choice: `.red); } else { stdin.pause(); if (stats[Number(data)].isDirectory()) { fs.readdir( path.resolve(__dirname, filename), function(err, files) { console.log(''); console.log(`(${files.length} files)`); files.forEach(function(file) { console.log(` - ${file}`); }); } ); } else { fs.readFile( path.resolve(__dirname, filename), 'utf8', function (err, data) { console.log(''); console.log(data.cyan); } ); } } } file(0);
###
fs升级demo 升级需求
程序在命令行运行。通过终端提供交互给用户输入,输出。
程序启动后,需要显示当前目录下的文件列表。
选择某个文件时,程序需要显示该文件的内容。
选择一个目录时,程序需要显示该目录下的文件列表,且能一直递归下去
运行结束后退出程序。
code 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 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 var fs = require('fs'), colors = require('colors'), stdin = process.stdin, stdout = process.stdout, path = require('path'), stats = [], cwd = process.cwd(), files = fs.readdirSync( cwd ); function file(i) { var filename = files[i]; fs.stat( path.resolve(cwd, filename), function (err, stat) { stats[i] = stat; if( stat.isDirectory() ) { console.log(` ${i} ${filename}/`.blue); } else { console.log(` ${i} ${filename}`.green); } if( ++i == files.length ) { read(); } else { file(i); } } ); } function read() { console.log(``); stdout.write(` Enter your choice: `.yellow); stdin.resume(); stdin.setEncoding('utf8'); stdin.removeListener('data', option); stdin.on('data', option); } function option(data) { var filename = files[Number(data)]; if( !filename ) { stdout.write(` Enter your choice: `.red); } else { stdin.pause(); if( stats[Number(data)].isDirectory() ) { cwd = path.resolve(cwd, filename); fs.readdir( cwd, function(err, subFiles) { files = subFiles; file(0, cwd); } ); } else { fs.readFile( path.resolve(cwd, filename), 'utf8', function(err, data) { console.log(''); console.log(data.cyan); } ); } } } file(0);