虽然单线程简化了编程代码,因为这样咱们不必太担心并发引出的问题,这也意味着在阻塞主线程的情况下执行长时间的操作,如网络请求。...使用异步 (如 回调函数、promise、async/await),可以不用阻塞主线程的情况下长时间执行网络请求。 了解异步的工作方式之前,咱们先来看看同步是怎么样工作的。...最后,当networkRequest()函数完成时,调用greeting()函数。 因此,咱们必须等待函数如processImage()或networkRequest()完成。...等待某个事件(在本例中单击event)发生,当该事件发生时,回调函数被放置在等待执行的消息队列中。...0秒后,bar()回调被放入等待执行的消息队列中,但是它只会在堆栈完全空的时候执行,也就是在baz和foo函数完成之后。
可以继续使用浏览器发起http请求来查看结果,一个简单且熟悉的JSON数据接口已经完成了。你可以按照前端最常用的调用方式,如ajax或者axios来请求接口来在你的前端项目使用它。...; }) 上面的例子中,新使用Nodejs另一个自带的模块url,url模块顾名思义是一个处理href的库,它将href拆分成各个子内容,同时为了能处理客户端带来的userId参数还使用了自带的querystring...一个规范的接口已经开发完成了,简单总结一下,上面以渐进式的方式了解了Nodejs如何给前端来提供一个规范化的http接口,了解了服务端的JavaScript所具备的能力,接下来再从另一个前端工程师比较少接触的内容...1.读一个文件 在Nodejs中,读文件有两种形式,一种是同步的另一种是异步的,同步可以理解为读文件这个过程要等待,就是一旦执行的读这个操作的时候,你的代码就被”卡“住了,直到文件读完才能继续执行,来看看下面的例子...图片上半部分是异步读文件,可以看出来,读的操作可以理解为同时刻执行的。 图片下半部分是同步读文件,第二个读的操作需要先等待第一个读完才可以。
这样就把等待的时间充分利用了起来,提高了执行效率,但是同时又会产生一个问题,nodejs 应用程序怎么知道操作系统已经完成了 I/O 操作呢?...对于轮询而言,有以下这么几种方案: 一直轮询检查I/O状态,直到 I/O 完成。这是最原始的方式,也是性能最低的,会让 CPU 一直耗用在等待上面。其实跟阻塞 I/O 的效果是一样的。...nodejs 发一个完成信号,nodejs 执行回调操作。...创建请求对象 以Windows系统为例来说,在这个函数的调用过程中,我们创建了一个文件I/O的请求对象,并往里面注入了回调函数。...属性对应的值便是我们 nodejs 应用程序代码中传入的回调函数。
所以如果JS执行的时间太长,则会造成页面卡顿的情况,严重影响用户体验。如果GUI线程被JS引擎执行所阻塞时,所有GUI更新将会被提交到一个队列中,直到JS引擎空闲时才会释放GUI线程。...当js执行碰到事件绑定和一些异步操作(如setTimeOut,也可来自浏览器内核的其他线程,如鼠标点击、AJAX异步请求等),会走事件触发线程将对应的事件添加到对应的线程中(比如定时器操作,便把定时器事件添加到定时器线程...简单说就是当执行到一个http异步请求时,就把异步请求事件添加到异步请求线程,等收到响应(准确来说应该是http状态变化),再把回调函数添加到事件队列,等待js引擎线程来执行宏任务与微任务了解JavaScript...在JavaScript中任务大致上分为两种任务:同步任务:可以等同与无异步逻辑的异步任务。顺序执行,与其他语言的同步任务相同。...NodeJS中的运行机制 Node 中的 Event Loop 和浏览器中的是完全不相同的东西。
挂起函数的概念 挂起函数是具有suspend关键字修饰的函数,它可以在协程内部被挂起,等待某个操作完成后再继续执行。典型的例子包括网络请求、文件读写、数据库查询等异步操作。...函数)可以在挂起函数内部创建新的协程,它会等待所有的子协程完成后再继续执行。...通过使用launch函数,我们可以在不同的协程中同时执行多个任务,而这些协程可以在相同的作用域内运行,继承相同的上下文和调度器。...协程提供了async函数来实现这种顺序性操作,通过await等待前一个操作的完成。...例如,一个协程可以等待另一个协程发送特定的信号,或者通过关闭通道来取消一个协程。
在常规的服务器端程序设计中, 比如说爬虫程序, 发送http请求的过程会使整个执行过程阻塞,直到http请求响应完成代码才会继续执行, 以php为例子 当代码执行到第二行时,程序便陷入了等待,直到请求完成...以最简单的前端ajax请求为例 代码先输出1,再输出2,整个程序执行流程并未因http请求而被阻塞,回调函数方案完美的把问题解决。 然而,这只是最简单回调函数示例,假如回调函数嵌套了许多层呢?...因为没辙啊, 试想一下,ajax的回调函数中使用return语句, 意义何在?因此也只能变向的通过Promise将返回值扔给外部的调用者。...换句话说, resolve和reject的调用是用来通知await等待结束,代码可以继续执行了。 这种写法不就是之前想方设法想实现的同步写法么?...所以, 第二个要点就是 await就是用来等待Promise对象中resolve和reject这两个函数的执行的,并且将这两个函数传递的参数当作返回结果赋给变量,如同run函数中的代码示例那样。
JavaScript中几乎所有的I/O基元都是非阻塞的,如网络请求、文件系统操作等。...()未完成),因此它将等待剩余的毫秒数,直到达到最快计时器的时间点。...在等待95毫秒时,fs.readFile()完成读取文件,并将需要10毫秒才能完成的回调添加到轮询队列中并执行。...check 正常来说,在执行代码时,事件循环最终将进入poll阶段,在该阶段,它将等待传入连接、请求等。...callbacks完成一个执行周期,其中的poll用来处理异步操作 参考资料 https://zhuanlan.zhihu.com/p/33058983 http://nodejs.cn/learn/
主要工作在CPU进行)I/O(如读写文件、读写数据库、读写网络请求等。...主要工作在各种I/O设备,如磁盘、网卡等)一个典型的传统web应用实现多进程,一个请求fork一个(子)进程 + 阻塞I/O(即blocking I/O或BIO)多线程,一个请求创建一个线程 + 阻塞I...函数只有在内部工作全部执行完成后才会返回给调用者所以阻塞I/O是,应用程序通过API调用I/O操作后,当前进(线)程将会进入等待状态,代码无法继续往下执行,这时CPU可以进行进(线)程调度,即切换到其他可执行的进...I/O操作,通过API调用I/O操作后会马上返回,紧接着就可以继续执行其他代码逻辑,那为什么nodejs中的I/O是“非阻塞”的呢?...I/O请求,等待数据准备好(如网络I/O,等待数据从网络中到达socket;等待系统从磁盘上读取数据等)数据准备好后,复制到内核缓冲区从内核空间复制到用户空间,用户程序拿到数据接下来我们看一下操作系统中有哪些
这时事件触发的执行流程,比如函数等,将会进入回调的处理过程,而为了实现不同回调的实现,浏览器提供了一个消息队列。 当主线上下文内容都程执行完成后,会将消息队列中的回调逻辑一一取出,将其执行。...一种是setTimeout定时器作为代表的,触发后直接进入事件队列等待执行;一种是XMLHTTPRequest代表的,触发后需要调用去另一个线程执行,执行完成后封装返回值进入事件队列等待。...在宏任务执行过程中,v8引擎都会建立新栈存储任务,宏任务中执行不同的函数调用,栈随执行变化,当该宏任务执行结束时,会清空当前的栈,接着主线程继续执行下一个宏任务。...此时当前已没有主逻辑执行的代码,而当前宏任务将执行结束,微任务会在当前宏任务完成前执行,所以微任务队列会依次执行,直到微任务队列清空。...而通过查看Nodejs版本日志发现,在Nodejs环境中,在11版本之前,同源的任务放在一起进行执行,也就是宏任务队列和微任务队列只有清空一个后才会执行另一个。
主要工作在CPU进行)I/O(如读写文件、读写数据库、读写网络请求等。...主要工作在各种I/O设备,如磁盘、网卡等)一个典型的传统web应用实现多进程,一个请求fork一个(子)进程 + 阻塞I/O(即blocking I/O或BIO)多线程,一个请求创建一个线程 + 阻塞I...函数只有在内部工作全部执行完成后才会返回给调用者所以阻塞I/O是,应用程序通过API调用I/O操作后,当前进(线)程将会进入等待状态,代码无法继续往下执行,这时CPU可以进行进(线)程调度,即切换到其他可执行的进...I/O操作,通过API调用I/O操作后会马上返回,紧接着就可以继续执行其他代码逻辑,那为什么nodejs中的I/O是“非阻塞”的呢?...(如网络I/O,等待数据从网络中到达socket;等待系统从磁盘上读取数据等)数据准备好后,复制到内核缓冲区从内核空间复制到用户空间,用户程序拿到数据接下来我们看一下操作系统中有哪些I/O模型参考nodejs
交付被委托给一个异步任务调度程序离线完成。 在分布式系统中,许多任务是在用户的单个请求的背景下运行。考虑Facebook、WhatsApp 或 Instagram 这样的热门系统有数亿用户。...向客户发送电子邮件通知,告知其账户扣除一定金额的资金,是另一个需要紧急执行的任务示例。...所以,须考虑如何在非高峰时段更好利用资源及如何在高峰时段保持资源可用。 有些任务无需紧急执行。如Facebook社交应用,建议好友不是紧急任务。...我们要求任务是幂等的。幂等任务无论执行多少次都会产生相同的结果。 此属性是由开发人员在实现中添加的,通过某些内容(例如名称)来标识该属性并覆盖旧的。...设计中的每个组件都是分布式的,使得整个系统可用性大大增强。 8.2 持久性 我们将任务存储在持久化分布式数据库中,并在接近执行时间时将任务推送到队列中。一旦提交任务,它就会在数据库中直到执行完成。
如果说nodejs是单线程语言,可以想象一下,一个单实例的nodejs的服务器同时接受100个用户请求时,第100个用户的请求要等前面99的用户处理完成才能得到处理,如果每个用户的请求要0.3秒,第100...三. nodejs的异步IO 再拿上面的例子来看,当100个用户请求同时被接受到时,当需要IO(网络IO/文件IO)操作时,单线程的javascript并不会停下来等待IO操作完成,而是“事件驱动”开始介入...事件循环 说到事件循环,在上面的请求中,100个请求的都在非常短的时间得到了处理,而后请求又各自得到了回复,可以思考一下,javascript已经执行到了第100个请求,而第1个请求才得到回复,而第一个请求的栈信息没有丢失...,说明第一个请求的请求栈信息被记录了,这一过程便是注册IO事件。...可以思考一下,读操作是线程池来控制执行的,在该线程执行前,先在注册事件的内存中初始化一个状态是“执行中”,并且事件循环也已经被激活,开始轮询等待执行结果,当执行IO的线程在执行完之后,再通过底层的异步IO
入口文件被看作是同步代码,由v8来执行。在从上到下的解释/编译的过程中,如果遇到执行异步代码的请求的时候,nodejs就会把它交给event loop来执行。...之所以说它是“半无限”,是因为当没有任何任务(更多的异步I/O请求或者timer)要做的的时候,event loop会退出这个循环,整个nodejs程序也就执行完成了。...event loop会依次执行队列中的callback,直到队列为空,才会移步到下一个阶段。setImmediate()实际上是执行在另一个阶段的timer。...nodejs与browser中event loop的区别这两者之间有相同点,也有差异点。再次强调,以下结论是基于node v12.16.2来得出的。...只有等待时间超出了nodejs计算出来的限定时间或者再也没有未完成的I/O任务的时候,nodejs才会退出event loop。
Nodejs 中提供了 http 模块可以让 js 做相同的事。...然后异步调用的第一阶段就完成了,JavaScript 会继续往下执行执行栈上的代码逻辑,当前的 I/O 操作将以请求对象的形式放入到线程池中,等待执行。达到了异步 I/O 的目的。...第三阶段:事件循环中 I/O 观察者,会从请求对象中找到已经得到结果的 I/O 请求对象,取出结果和回调函数,将回调函数放入事件循环中,执行回调,完成整个异步 I/O 任务。...在 NodeJS 中不止一个队列,不同类型的事件在它们自己的队列中入队。在处理完一个阶段后,移向下一个阶段之前,事件循环将会处理两个中间队列,直到两个中间队列为空。...再检查当前队列的第一个,直到没有过期的,移到下一个阶段。
如果需要较长的时间来完成请求,则处理请求的线程会一直被占用,直到方法调用结束。不幸的是,多数较长的调用是由较长的数据库查询或对另一个 Web 服务的调用等事件引起的。...但结果是相同的,即释放线程以使其能够执行其他进程。 ASMX 处理程序调用服务器的 BeginXXX 函数后,会将线程返回到进程线程池,使之能够处理接收到的任何其他请求。...但是,还不能释放请求的 HttpContext。ASMX 处理程序将等待,直到它传递给 BeginXXX 函数的回调函数被调用,它才结束处理请求。 ...一旦回调函数被调用,ASMX 处理程序将调用 EndXXX 函数,使您的 Web 方法可以完成任何所要执行的处理,并且可以得到被序列化到 SOAP 响应中的返回数据。...然后,回调函数将调用 EndGetAge 方法以完成请求。此示例中的代码比前面示例中的代码简单得多,并且还具有另外一个优点,即没有在与为中间层 Web 方法请求提供服务的相同线程池中启动后端处理。
看过我之前的写的文章的朋友们应该会知道,使用NodeJS创建一个HTTP服务器是非常简单的。我们写的一个个API中使用req来接收请求,使用res来响应请求。...http的头部信息是如何进行创建呢?接下来我们来从源码角度分析:NodeJS究竟是如何处理一个HTTP请求。...我们可以看到在Server构造函数中设置了request和connection事件的回调函数: request使用了createServer中设置的回调方法requestListener。...而Stream是NodeJS另一个尤其重要的知识点,不过本篇文章不进行深入讲解。...5.最终回到listen()方法并且self.emit('connection', socket); 这样在对listen事件的调用中实现对端口的监听。到这里一个http请求就解析完成了。
领取专属 10元无门槛券
手把手带您无忧上云