Node.js 中阻塞、非阻塞及异步特性

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的异步特性

  1. 非阻塞I/O

例如发起文件读取、网络请求等耗时操作的操作的时候,不会等待结果返回,而且立刻执行后续代码;当耗时任务完成后,通过回调函数、Promise或者async/await处理结果(学习JavaScript的时候,你一定学习过回调地狱的问题,想要避免它,目前就可以Promise或async/await的方式来解决);

  1. 事件循环

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();

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

相关推荐
孜孜不倦不忘初心1 小时前
mac安装nvm及问题记录
前端·node.js
快乐的哈士奇5 小时前
Gmail-邮件自动处理系统
node.js·自动化·excel
星空6 小时前
Node.js (Express) + Vue2 Axios 前后端交互 CRUD
vue.js·node.js·express
云浪8 小时前
别再让用户干等了:用 Express + SSE 实现《红楼梦》AI 问答实时输出
javascript·后端·node.js
怕浪猫8 小时前
Electron 开发实战(十四):实战项目|从零搭建轻量化桌面代码编辑器
前端·electron·node.js
zhuxiaojt1 天前
npx 为何如此之慢?浅谈 npx 速度慢的原因及工具推荐
node.js
码农刚子1 天前
从零开始:在 Windows 服务器上部署 Node.js 项目(小白实战教程)
后端·node.js
MageGojo1 天前
用 Node.js 把聚合 API 平台封装成零依赖命令行工具:registry 驱动的工程实践
node.js·restful·api接口·命令行工具·cli
濮水大叔1 天前
浅论CabloyJS全栈框架提供的“两级页签”机制
typescript·node.js·next.js
meilindehuzi_a1 天前
深入理解 Ajax 异步请求:从 XMLHttpRequest 到 Node.js HTTP 服务实践
http·ajax·node.js