Node.js 中阻塞、非阻塞及异步特性
前言
- 看一下我们之前写的代码
vue
const fs = require('fs')
const textIn = fs.readFileSync('./txt/input.txt','utf-8')
console.log(textIn)
这个有什么问题呢?这串代码是同步的,同步的意思就是你必须等待上一行代码完全运行完毕之后你才可以运行下一行代码;这就是意味着如果你这行代码发生问题,这行代码就会阻塞下一行代码的运行,这样并不是好的办法。
- 所以我们可以采用异步的方式,如下
vue
const fs = require('fs');
fs.readFile('./txt/input.txt','utf-8',(err,data)=>{
console.log(data)
});
console.log('文件正在读取中......');

我们发现这样就不会导致代码异常来导致阻塞了;
Node.js的异步特性
- 非阻塞I/O
例如发起文件读取、网络请求等耗时操作的操作的时候,不会等待结果返回,而且立刻执行后续代码;当耗时任务完成后,通过回调函数、Promise或者async/await处理结果(学习JavaScript的时候,你一定学习过回调地狱的问题,想要避免它,目前就可以Promise或async/await的方式来解决);
- 事件循环
node.js单线程可以并发处理大量请求的原因就是使用事件循环不断检查任务队列,例如一些定时器、I/O请求、微任务等,按照阶段依次运行;
回调并不能代表异步,这些我们要自己去实现,而node.js中的像readfile这样的是Node实现的,我们需要使用它就自带这个特性
实际演示
javascript
fs.readFile('./txt/start1.txt','utf-8',(err,data) => {
if(err) return console.log('发生了错误');
console.log(data);
})
- 这个文本只有一个内容,read-this,而有一个read-this.txt里面有实际内容
vue
fs.readFile('./txt/start.txt','utf-8',(err,data) => {
if(err) return console.log('发生了错误');
console.log(data);
fs.readFile(`./txt/${data}.txt`,'utf-8',(err,data1) => {
console.log(data1)
})
})

- 紧接着,我们在读取一个文档
vue
fs.readFile('./txt/start.txt','utf-8',(err,data) => {
if(err) return console.log('发生了错误');
console.log(data);
fs.readFile(`./txt/${data}.txt`,'utf-8',(err,data1) => {
console.log(data1)
fs.readFile('./txt/append.txt','utf-8',(err,data2)=>{
console.log(data2)
})
})
})

- 继续折腾,创建一个文档,合并一下上面两个文档
vue
fs.readFile('./txt/start.txt','utf-8',(err,data) => {
if(err) return console.log('发生了错误');
console.log(data);
fs.readFile(`./txt/${data}.txt`,'utf-8',(err,data1) => {
console.log(data1)
fs.readFile('./txt/append.txt','utf-8',(err,data2)=>{
console.log(data2)
fs.writeFile('./txt/final.txt',`${data1}\n${data2}`,'utf-8',err => {
console.log('文件成功写入!');
})
})
})
})

- 上面就是node.js使用回调函数来完成异步特性的,但是是不是看上去非常难以理解,现代我们会使用Promise或者async/await来解决这个问题,让其看起来更加明朗;
javascript
const fs = require('fs').promises;
async function processFiles() {
try{
//1.读取start.txt,获取文件名
const fileName = await fs.readFile('./txt/start.txt','utf-8');
console.log(fileName);
//2.并行读取两个文件
const [data1,data2] = await Promise.all([
fs.readFile(`./txt/${fileName}.txt`,'utf-8'),
fs.readFile(`./txt/append.txt`,'utf-8')
]);
console.log(data1);
console.log(data2);
//3.合并内容,写入文件
await fs.writeFile('./txt/final.txt',`${data1}\n${data2}`,'utf8');
console.log('文件成功写入!');
} catch (err) {
console.log('发生了错误',err.message);
}
}
processFiles();

这样从上到下看下来,是不是逻辑非常的清晰