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

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

相关推荐
. . . . .14 小时前
Node.js 模块系统
node.js
我叫汪枫1 天前
在后台管理系统中,如何递归和选择保留的思路来过滤菜单
开发语言·javascript·node.js·ecmascript
技术钱1 天前
nodejs使用RustFS容器服务搭建对象存储oss
node.js·egg
大貔貅喝啤酒1 天前
接口测试_Postman(详细版)
javascript·测试工具·node.js·自动化·postman
桜吹雪1 天前
Langchain.js官方文档:构建具备按需加载技能的 SQL 助手
javascript·人工智能·node.js
拾贰_C1 天前
【node.js | Ubuntu | update】如何升级旧的nodejs本版至最新;如何升级npm
ubuntu·npm·node.js
湖边看客2 天前
在 Windows PowerShell 里给 Node.js 设置内存上限
node.js
zhensherlock2 天前
Protocol Launcher 系列:Beorg 高效任务管理的协议支持
前端·javascript·typescript·node.js·自动化·github·js
shao9185162 天前
第3章(2)——使用Gradio JavaScript Client
javascript·node.js·cdn·gradio·job·events·playcode
Rabbit_QL2 天前
npm 不是“前端的包管理器“—它是 Node.js 的
前端·npm·node.js