const fs = require("fs");
const path = require("path");
const filename = path.resolve(__dirname, "./temp/abc.txt");
const ws = fs.createWriteStream(filename, {
flags: //操作文件方式
encoding: "utf-8",//编码方式
highWaterMark: 16 * 1024//最多写入字节数
});
ws.on(事件名,处理函数)
ws.open 打开事件
ws.close 关闭事件
ws.error 错误事件
具体功能和解释可见上一篇 node-可读流 基本操作方法一致
ws.write(data)
代表写入一组数据 data可以是字符串或者buffer 同时会返回一个布尔值
true:写入通道没有被填满,接下来可以直接写入 无需排队(根据highWaterMark)
false:写入通道已经填满,接下来数据将进入写入队列(此时需注意背压问题,写入内存中的数据是有限的)
drain
当写入队列清空时会触发drain事件,用法举例
//当highWaterMark为3时,代表写入通道最大为3
function write() {
let flag = true;
while (i<=1024*10 && flag){
flag = ws.wtite('a');
i++;
}
}
write()//此时写入3个a之后 写入队列满 会返回flase 停止写入
ws.on('drain', ()=>{
write()//当写入队列处理完成之后会触发drain事件 在执行 就会重复在写入highWaterMark个 之后写满循环
}
ws.end([data])
代表结束写入 ,讲自动关闭文件
- 是否自动关闭取决于autoClose配置
- 配置默认为true
- data 是可选的,表示关闭前最后一次写入
rs.pipe
在讲解pipe之前 先说一下文件的复制和写入 从a–>b文件中
首先是最简单的暴力方法 不考虑内存与硬盘空间占用等
const fs = require("fs");
const path = require("path");
//方式1
async function method1() {
const from = path.resolve(__dirname, "./temp/abc.txt");
const to = path.resolve(__dirname, "./temp/abc2.txt");
console.time("方式1");
const content = await fs.promises.readFile(from);
await fs.promises.writeFile(to, content);
console.timeEnd("方式1");
console.log("复制完成");
}
方式2
通过写入流写出流 缓慢写入 对于内存占用很少 一边读一边写
async function method2() {
const from = path.resolve(__dirname, "./temp/abc.txt");
const to = path.resolve(__dirname, "./temp/abc2.txt");
console.time("方式2");
const rs = fs.createReadStream(from);
const ws = fs.createWriteStream(to);
rs.on("data", chunk => {
//读到一部分数据
const flag = ws.write(chunk);
if (!flag) {
//表示下一次写入,会造成背压
rs.pause(); //暂停读取
}
});
ws.on("drain", () => {
//可以继续写了
rs.resume();
});
rs.on("close", () => {
//写完了
ws.end(); //完毕写入流
console.timeEnd("方式2");
console.log("复制完成");
});
}
pipe
相当于直接吧可读流和可写流直接拼接在一起 替我们省略以上步骤
const rs = fs.createReadStream(from);
const ws = fs.createWriteStream(to);
rs.pipe(ws);
rs.on("close", () => {
console.timeEnd("方式2");
});
}