深入Nodejs模块fs
标签:
node 的fs文档密密麻麻的 api 非常多,终究全面撑持对文件系统的操纵。文档组织的很好,操纵基天职为文件操纵、目录操纵、文件信息、流这个风雅面,编程方法也撑持同步、异步和 Promise。
本文记录了几个文档中没详细描写的问题,可以更好地串联fs文档思路:
文件描述符
同步、异步与 Promise
目录与目录项
文件信息
stream
文件描述符文件描述符是一个非负整数。它是一个索引值,操纵系统可以按照它来找到对应的文件。
在 fs 的很多底层 api 中,需要用到文件描述符。在文档中,描述符凡是用fd来代表。例如:fs.read(fd, buffer, offset, length, position, callback)。与这个 api 相对应的是:fs.readFile(path[, options], callback)。
因为操纵系统对文件描述符的数量有限制,因此在结束文件操纵后,别忘记 close:
const fs = require("fs"); fs.open("./db.json", "r", (err, fd) => { if (err) throw err; // 文件操纵... // 完成操纵后,封锁文件 fs.close(fd, err => { if (err) throw err; }); }); 同步、异步与 Promise所有文件系统的 api 都有同步和异步两种形式。
同步写法不保举使用同步 api,会梗阻线程。
try { const buf = fs.readFileSync("./package.json"); console.log(buf.toString("utf8")); } catch (error) { console.log(error.message); } 异步写法异步写法写起来容易进入回调地狱。
fs.readFile("./package.json", (err, data) => { if (err) throw err; console.log(data.toString("utf8")); }); (保举)Promise 写法在 node v12 之前,需要本身借助 promise 封装:
function readFilePromise(path, encoding = "utf8") { const promise = new Promise((resolve, reject) => { fs.readFile(path, (err, data) => { if (err) return reject(err); return resolve(data.toString(encoding)); }); }); return promise; } readFilePromise("./package.json").then(res => console.log(res));在 node v12 中,引入了 fs Promise api。它们返回 Promise 东西而不是使用回调。 API 可通过 require(‘fs‘).promises 访谒。如此一来,开发本钱更低了。
const fsPromises = require("fs").promises; fsPromises .readFile("./package.json", { encoding: "utf8", flag: "r" }) .then(console.log) .catch(console.error); 目录与目录项fs.Dir 类:封装了和文件目录相关的操纵
fs.Dirent 类:封装了目录项的相关操纵。例如判断设备类型(字符、块、FIFO 等)。
它们之间的关系,通过代码展示:
const fsPromises = require("fs").promises; async function main() { const dir = await fsPromises.opendir("."); let dirent = null; while ((dirent = await dir.read()) !== null) { console.log(dirent.name); } } main(); 文件信息fs.Stats 类:封装了文件信息相关的操纵。它在fs.stat()的回调函数中返回。
fs.stat("./package.json", (err, stats) => { if (err) throw err; console.log(stats); });注意,关于查抄文件是否存在:
不建议在挪用 fs.open()、 fs.readFile() 或 fs.writeFile() 之前使用 fs.stat() 查抄文件是否存在。而是应该直接打开、读取或写入文件,如果文件不成用则措置惩罚惩罚引发的错误。
要查抄文件是否存在但随后并不同错误其进行操纵,则建议使用 fs.access()。
ReadStream 与 WriteStream在 nodejs 中,stream 是个非常重要的库。很多库的 api 都是基于 stream 来封装的。例如下面要说的 fs 中的 ReadStream 和 WriteStream。
fs 自己供给了 readFile 和 writeFile,它们好用的价钱就是性能有问题,会将内容一次全部载入内存。但是对付几 GB 的大文件,显然会有问题。
那么针对大文件的解决方案自然是:一点点读出来。这就需要用到 stream 了。以 readStream 为例,代码如下:
const rs = fs.createReadStream("./package.json"); let content = ""; rs.on("open", () => { console.log("start to read"); }); rs.on("data", chunk => { content += chunk.toString("utf8"); }); rs.on("close", () => { console.log("finish read, content is:\n", content); });借助 stream 的 pipe,一行快速封装一个大文件的拷贝函数:
function copyBigFile(src, target) { fs.createReadStream(src).pipe(fs.createWriteStream(target)); } 参考链接文件描述符
Socket 套接字
Nodejs 根本:stream 模块入门介绍与使用
Fastest way to copy file in node.js
Using Node.js to Read Really, Really Large Datasets & Files (Pt 1)
最后感受不错,资助点个保举呗,您的撑持是对我最大的激励
温馨提示: 本文由Jm博客推荐,转载请保留链接: https://www.jmwww.net/file/web/30915.html
