Nodejs - 异步I/O

异步I/O

利用单线程,远离多线程死锁,状态同步等问题,利用异步I/O, 让单线程原理阻塞,更好的使用cpu

在这里插入图片描述

异步I/O实现现状

  1. 阻塞IO
    操作系统内对于I/O只有两种方式: 阻塞和非阻塞。在调用阻塞I/O的时候,应用程序需要等待I/O完成之后才返回结构。
    阻塞I/O的特点是调用之后等到系统内核层面完成所有操作之后,调用才结束。阻塞i/o造成CPU等待I/O,浪费等待事件,CPU的处理能力不能被充分利用
    在这里插入图片描述

  2. 非阻塞I/O
    操作系统对计算机进行了抽象,将所有的输入输出设备抽象为文件,内核在进行文件I/O的操作的时候,通过文件操作符进行管理,而文件描述符类似于引用程序和系统内核之间的凭证,应用程序如果需要对I/O进行调用,需要调用文件描述符。非阻塞I/O获取数据,不带数据进行返回,如果需要获取数据,需要调用文件操作符进行读取。为了获取完整的数据,引用程序需要重复调用I/O操作来确认是否完成。
    在这里插入图片描述

    非阻塞I/O返回的后,CPU的时间片可以用来处理其他事务,

    • 轮询
      减少I/O状态判断的CPU损耗
      1. read 重复调用检查I/O的状态来完成数据的读取,在获得到最终的数据之前,cpu会一直耗用在等待上。
        在这里插入图片描述

      2. select 在read的基础上进行改进的一种方案,通过对文件描述符上的事件状态来判断。select轮询必须使用1024数组来存储状态。
        在这里插入图片描述

      3. poll 使用链表的方式避免数组长度限制,其次避免不需要的检查。当文件操作符较多的时候,性能还是十分低下的。
        在这里插入图片描述

      4. epoll linux下最高效率的I/O事件通知机制。进入轮询的时候,如果没有检查到的I/O事件。就会进行休眠,知道事件将它唤醒,利用事件通知,执行回调的方式,而不是遍历查询
        在这里插入图片描述

      5. kqueue 仅仅在FreeBSD系统下运行,类似于epoll

  3. 异步I/O
    Node在*nix平台下实现了libeio配合libev实现io部分。采用了线程池和阻塞I/O模拟异步。
    在window利用IOCP
    区别: node提供了libuv作为抽象封装层。是的所有的平台兼容性判断都有这一层来完成,保证上层的node和下层的自定义线程池之间独立。
    在node是单线程,这里的单线程仅仅是JavaScript执行在单线程
    在这里插入图片描述

Node的异步I/O

  1. 事件循环
    node自身的执行模型- 事件循环。
    在进程启动的时候,会创建一个while的循环,每执行一个循环体的过程称为tick,每个tick的过程就是查看是否有是事件待处理,如果有,就取出事件以及相关的回调函数。如果存在关联的回调函数。就执行。无的话,退出流程。
    

在这里插入图片描述

  1. 观察者
    每个事件循环中有一个或者多个观察者,判断是否有事件要处理就是向观察者询问是否有要处理的事件。
    node中,事件主要来源于网路请求,文件I/O, 这些事件对应的观察者都有文件I/O观察者,网络I/O观察者。
    事件循环是一个生产者/消费者模型。生产者: 异步io,网络请求。事件传递到观察者,事件循环从观察者那边取出事件循环。

  2. 请求对象

    fs.open = function(path, flags, mode, callback) {
    binding.open(pathModule._makeLong(path), stringToFlags(flags), mode, callback)
    }
    

    fs.open() 根据执行路径和参数打开一个文件,从而得到一个文件操作符,这是后续io操作的初始操作。
    在这里插入图片描述

    从JavaScript中调用node 的核心模块,核心模块中调用c++内建模块,内建模块通过libuv进行系统调用,这里的libuv作为封装层,有两个平台的实现,实际上是调用了uv_fs_open()的方法,在uv_fs_open()的调用过程中,创建了一个FSReqWrap请求对象,从JavaScript传入的参数和当前方法都被封装在这个请求对象中,最为关注的回调函数则被设置在这个对象的oncomplete_sym属性上。

    • 请求对象是异步io中的重要中间产物,所有的状态都保存在这个对象中,包括送入线程和io操作完毕之后的回调处理
  3. 执行回调
    组装好请求对象,送入io线程池等待执行, 实际上完成了异步io第一部分
    线程池中的io操作调用完毕之后,会将获取的对象存储在req-result属性上。
    在这里插入图片描述

    事件循环,观察者,请求对象,io线程池这四者共同构成了node异步io模型的四要素

非io的异步api

  • setTimeout() setInterval()
    分别用于单次和多次定时执行任务。 实现的原理和异步io比较相似,只是不需要io线程池的参与。
    调用setTimeout()和setInterval() 创建的迭代器会被插入到是定时器观察者内部的一个红黑树中,每次tick执行,会从该红黑树中迭代取出定时器对象,检查是否超过定时时间。如果超过就形成一个事件,回调函数会立即执行
    定时器的问题在于,并非精确的在容忍事件范围内,尽管事件循环十分快,但是如果某次循环占用的时候过多,下次循环的时候,也会超时很久。
    在这里插入图片描述

  • process.nextTick()

    process.nextTick = function(callback) {if(process._exiting) return if(tickDepth >= process.maxTickDepth)maxTickWarn()var tock = { callback: callback } if (process.domain) tock.domain = process.domainnextTickQueue.push(tock)if (nextTickQueue.length) {process._needTickCallback()}
    }
    
  • setImmediate()
    setImmediate()和process.nextTick()方法十分类似,都是将回调函数延迟执行

    process.nextTick(function() {console.log("延迟执行")
    })
    console.log("正常执行")
    // 正常执行 延迟执行
    
    setImmediate(function() {console.log('延迟执行')
    })
    console.log("正常执行")
    // 正常执行 延迟执行
    
    process.nextTick(function () {console.log('nextTick延迟执行')
    })
    setImmediate(function () {console.log('setImmediate延迟执行')
    })
    console.log('正常执行')
    // 正常执行
    // nextTick 延迟执行
    // setImmediate 延迟执行
    

    process.nextTick()中的回调函数执行的优先级要高于setImmediate() 时间循环对于观察者的检查是由先后顺序的,process.nextTick()属于idle观察者。setImmediate()属于check观察者,在每一次循环检查中,idle观察者先于io观察者,io观察者先于check观察者、
    process.nextTick() 的回调函数保存在一个数组中,setImmediate()的结果保存在链表中。process.nextTick()在每轮循环中会将数组中的回调函数全部执行完,setImmediate()在每轮循环中执行链表中的一个回调函数

    process.nextTick(function() {console.log('nextTick延迟执行1')
    })
    process.nextTick(function()) {console.log('nextTick延迟执行2')
    }
    setImmediate(function() {console.log('setImmediate延迟执行1')process.nextTick(function () {console.log('强势插入')})
    })
    setImmediate(function() {console.log('setImmediate延迟执行2')
    })
    console.log('正常执行')
    // 正常执行
    // nextTick延迟执行1
    // nextTick延迟执行2
    // setImmediate延迟执行1
    // 强势插入
    // setImmediate延迟执行2
    

在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/637758.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

能源成果3D网络三维展厅越发主流化

在这个数字化飞速发展的时代,我们为您带来了全新的展览形式——线上3D虚拟展厅。借助VR虚拟现实制作和web3d开发技术,我们能够将物品、图片、视频和图文信息等完美融合,通过计算机技术和3D建模,为您呈现一个逼真、生动的数字化展览…

Llama 3大模型发布!快速体验推理及微调

Meta,一家全球知名的科技和社交媒体巨头,在其官方网站上正式宣布了一款开源的大型预训练语言模型——Llama-3。 据了解,Llama-3模型提供了两种不同参数规模的版本,分别是80亿参数和700亿参数。这两种版本分别针对基础的预训练任务…

JDBC学习

DriverManager(驱动管理类) Drivermanager的作用有: 1.注册驱动; 2.获取数据库连接 Class.forName("com.mysql.cj.jdbc.Driver"); 这一行的作用就是注册Mysql驱动(把我们下载的jar包加载到内存里去&…

地图图源#ESRI ArcGIS XYZ Tiles系列(TMS)

目录 1、前言 2、地图图源网址 2.1、Satellite 卫星图源 2.2、Terrain 地形图源 2.3、Street 路网/标注图源 2.4、Specifity 特色设计图源 3、专业推荐”穿搭“ 4、图源配置下载及使用 图源名称图层类别特别注意谷歌 Google①地形 ②影像 ③矢量及标注 ④特色图源国内大…

【讲解下Spring Boot单元测试】

🌈个人主页: 程序员不想敲代码啊 🏆CSDN优质创作者,CSDN实力新星,CSDN博客专家 👍点赞⭐评论⭐收藏 🤝希望本文对您有所裨益,如有不足之处,欢迎在评论区提出指正,让我们共…

HotSpot JVM 中的应用程序/动态类数据共享

0.前言 本文的目的是详细讨论 HotSpot JVM 自 JDK 1.5 以来提供的一项功能,该功能可以减少启动时间,但如果在多个 JVM 之间共享相同的类数据共享 (CDS) 存档,则还可以减少内存占用。 1.类数据共享 (CDS) CDS 的想法是使用特定格式将预处理…

【LeetCode】187. 重复的DNA序列

题目链接:187. 重复的DNA序列 题目描述: 思路:首先要明白一个重要概念区别: 子串(substring): 原始字符串的一个连续子集。子序列(subsequence): 原始字符串的一个子集。 再注意题意是:长度为10的子串出现…

强固型工业电脑在码头智能化,龙门吊/流机车载电脑的行业应用

码头智能化行业应用 对码头运营来说,如何优化集装箱从船上到码头堆场到出厂区的各个流程以及达到提高效率。 降低成本的目的,是码头营运获利最重要的议题。为了让集装箱码头客户能够安心使用TOS系统来调度指挥码头上各种吊车、叉车、拖车和人员&#xf…

安居水站:理解烟瘾、探索戒烟动因及采用有效策略

摘要:本论文旨在深入探讨烟瘾的成因,分析戒烟的动因和好处,并提出一套科学的戒烟方法和具体策略。通过综合权威机构的研究文献,我们将为公众提供更全面、更深入的理解,从而推动更多人走上戒烟的道路,改善个…

鸿蒙OpenHarmony【轻量系统编写“Hello World”程序】 (基于Hi3861开发板)

编写“Hello World”程序 下方将通过修改源码的方式展示如何编写简单程序,输出“Hello world”。请在下载的源码目录中进行下述操作。 前提条件 已参考鸿蒙开发指导文档:gitee.com/li-shizhen-skin/harmony-os/blob/master/README.md点击或者复制转到…

力扣283. 移动零

Problem: 283. 移动零 文章目录 题目描述思路复杂度Code 题目描述 思路 1.定义一个int类型变量index初始化为0; 2.遍历nums当当前的元素nums[i]不为0时使nums[i]赋值给nums[index]; 3.从index开始将nums中置对应位置的元素设为0; 复杂度 时间…

从零开始的vscode配置及安装rust教程

配置vscode的rust环境 下载安装vscodemac 环境1. 下载安装rust2. 配置 mac vscode环境3. 创建一个测试项目 windows 环境1. 安装c运行环境2. 安装配置rustup3. 配置windows vscode环境4. 创建一个测试项目 下载安装vscode 1.官网应用程序下载 vscode:https://code.v…