成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

Node.js 的微任務處理(基于Node.js V17)

開發 前端
Node.js 的事件循環已經老生常談,但是在 Node.js 的執行流程中,事件循環并不是全部,在事件循環之外,微任務的處理也是核心節點.

[[441996]]

前言:Node.js 的事件循環已經老生常談,但是在 Node.js 的執行流程中,事件循環并不是全部,在事件循環之外,微任務的處理也是核心節點,比如 nextTick 和 Promise 任務的處理。本文介紹 Node.js 中微任務處理的相關內容。網上文章和很多面試題中有很多關于 Promise、nextTick、setTimeout 和 setImmediate 執行順序的內容。通過本文,讓你從原理上理解他們,碰到相關的問題就引刃而解,不再拘泥于背誦和記錄。

1 事件循環

本文不打算詳細地講解事件循環,因為已經有很多相關文章,而且本身也不是很復雜的流程。事件循環本質上是一個消費者和生產者的模型,我們可以理解事件循環的每一個階段都維護了一個任務隊列,然后在事件循環的每一輪里就會去消費這些任務,那就是執行回調,然后在回調里又可以生產任務,從而驅動整個事件循環的運行。當事件循環里沒有生產者的時候,系統就會退出。而有些生產者會 hold 住事件循環從而讓整個系統不會退出,比如我們啟動了一個 TCP 服務器。事件循環處理了 Node.js 中大部分的執行流程,但是并不是全部。

2 微任務

Node.js 中,典型的微任務包括 nexiTick 和 Promise。官網說 nextTick 任務會在繼續事件循環之前被處理,描述得比較宏觀,下面我們來看一下具體的實現細節。微任務的處理時機分為兩個時間點。1. 定義 C++ InternalCallbackScope 對象,在對象析構時。2. 主動調 JS 函數 runNextTicks。

2.1 InternalCallbackScope

下面先看一下 InternalCallbackScope。通常在需要處理微任務的地方定義一個 InternalCallbackScope 對象,然后執行一些其他的代碼,最后退出作用域。

  1.     InternalCallbackScope scope 
  2.     // some code 
  3.  
  4. } // 退出作用域,析構 

下面看一下 InternalCallbackScope 析構函數的邏輯。

  1. InternalCallbackScope::~InternalCallbackScope() { 
  2.   Close(); 
  3.  
  4.  
  5. void InternalCallbackScope::Close() { 
  6.  
  7.  tick_callback->Call(context, process, 0, nullptr); 
  8.  

在析構函數里會執行 tick_callback 函數。我們看看這個函數是什么。

  1. static void SetTickCallback(const FunctionCallbackInfo<Value>& args) { 
  2.   Environment* env = Environment::GetCurrent(args); 
  3.   CHECK(args[0]->IsFunction()); 
  4.   env->set_tick_callback_function(args[0].As<Function>()); 
  5.  

tick_callback 是由 SetTickCallback 設置的。

  1. setTickCallback(processTicksAndRejections); 

我們可以看到通過 setTickCallback 設置的這個函數是 processTicksAndRejections。

  1. function processTicksAndRejections() { 
  2.   let tock; 
  3.   do { 
  4.     while (tock = queue.shift()) { 
  5.       const callback = tock.callback; 
  6.       callback(); 
  7.     } 
  8.     runMicrotasks(); 
  9.   } while (!queue.isEmpty() || processPromiseRejections()); 
  10.  

processTicksAndRejections 正是處理微任務的函數,包括 tick 和 Promise 任務。現在我們已經了解了 InternalCallbackScope 對象的邏輯。那么下面我們來看一下哪里使用了這個對象。第一個地方是在 Node.js 初始化時,執行完用戶 JS 后,進入事件循環前。看看相關代碼。

我們看到在 Node.js 初始化時,執行用戶 JS 后,進入事件循環前會處理一次微任務,所以我們在自己的初始化 JS 里調用了 nextTick 的話,就會在這時候被處理。第二個地方是每次從 C、C++ 層執行 JS 層回調時。

  1. MaybeLocal<Value> AsyncWrap::MakeCallback(const Local<Function> cb, 
  2.                                           int argc, 
  3.                                           Local<Value>* argv) { 
  4.   ProviderType provider = provider_type(); 
  5.   async_context context { get_async_id(), get_trigger_async_id() }; 
  6.   MaybeLocal<Value> ret = InternalMakeCallback( 
  7.       env(), object(), object(), cb, argc, argv, context); 
  8.   return ret; 
  9.  

MakeCallback 是 C、C++ 層回調 JS 層的函數,這個函數里又調用一個 InternalMakeCallback。

  1. MaybeLocal<Value> InternalMakeCallback(Environment* env, 
  2.                                        Local<Object> resource, 
  3.                                        Local<Object> recv, 
  4.                                        const Local<Function> callback, 
  5.                                        int argc, 
  6.                                        Local<Value> argv[], 
  7.                                        async_context asyncContext) { 
  8.  
  9.   // 定義 InternalCallbackScope 
  10.   InternalCallbackScope scope(env, resource, asyncContext, flags); 
  11.   // 執行 JS 層回調 
  12.   callback->Call(context, recv, argc, argv); 
  13.   // 處理微任務 
  14.   scope.Close(); 
  15.  

我們看到 InternalMakeCallback 里定義了一個 InternalCallbackScope,然后在回調完 JS 函數后會調用 InternalCallbackScope 對象的 Close 進行微任務的處理。

以上是典型的處理時機。另外在某些地方也會定義 InternalCallbackScope 對象,具體可在源碼里搜索。

2.2 runNextTicks

剛才介紹了每次事件循環消費任務時,就會去遍歷每一個階段的任務隊列,然后逐個執行任務節點對應的回調。執行回調的時候,就會從 C 到 C++ 層,然后再到 JS 層,執行完 JS 代碼后,會再次回調 C++ 層,C++ 層會進行一次微任務的處理,處理完后再回到 C 層,繼續執行下一個任務節點的回調,以此類推。這看起來覆蓋了所有的情況,但是有兩個地方比較特殊,那就是 setTimeout 和 setImmediate。其他的任務都是一個節點對應一個 C、C++ 和 JS 回調,所以如果在 JS 回調里產生的微任務,在回到 C++ 層的時候就會被處理。但是為了提高性能,Node.js 的定時器和 setImmediate 在實現上是一個底層節點管理多個 JS 回調。這里以定時器為例,Node.js 在底層使用了一個 Libuv 的定時器節點管理 JS 層的所有定時器,并在 JS 層里維護了所有的定時器節點,然后把 Libuv 定時節點的超時時間設置為 JS 層最快到期的節點的時間,這樣就會帶來一個問題。就是當有定時器超時,Libuv 從 C、C++ 回調 JS 層時,JS 層會直接處理所有的超時節點后再回到 C++ 層,這時候才有機會處理微任務。這會導致 setTimeout 里產生的微任務沒有在宏任務(setTimeout 的回調)執行完后被處理。這就不符合規范了。所以這個地方還需要特殊處理一下。我們看看相關的代碼。

  1. function processTimers(now) { 
  2.     nextExpiry = Infinity; 
  3.     let list; 
  4.     let ranAtLeastOneList = false
  5.     while (list = timerListQueue.peek()) { 
  6.       if (list.expiry > now) { 
  7.         nextExpiry = list.expiry; 
  8.         return refCount > 0 ? nextExpiry : -nextExpiry; 
  9.       } 
  10.       // 處理 listOnTimeout 最后一個回調里產生的微任務 
  11.       if (ranAtLeastOneList) 
  12.         runNextTicks(); 
  13.       else 
  14.         ranAtLeastOneList = true
  15.       listOnTimeout(list, now); 
  16.     } 
  17.     return 0; 
  18.  
  19. function listOnTimeout(list, now) { 
  20.     let ranAtLeastOneTimer = false
  21.     let timer; 
  22.     while (timer = L.peek(list)) { 
  23.       // 處理微任務 
  24.       if (ranAtLeastOneTimer) 
  25.         runNextTicks(); 
  26.       else 
  27.         ranAtLeastOneTimer = true
  28.       // 執行 setTimeout 回調 
  29.       timer._onTimeout(); 
  30.     } 
  31.  

定時器的架構如下。

Node.js 在 JS 層維護了一個樹,每個節點管理一個列表,處理超時事件時,就會遍歷這棵樹的每個節點,然后再遍歷這個節點對應隊列里的每個節點。而上面的代碼就是保證在每次調用完一個 setTimeout 回調時,都會處理一次微任務。同樣 setImmediate 任務也是類似的。

  1. let ranAtLeastOneImmediate = false
  2.  while (immediate !== null) { 
  3.    if (ranAtLeastOneImmediate) 
  4.      runNextTicks(); 
  5.    else 
  6.      ranAtLeastOneImmediate = true
  7.  
  8.   immediate._onImmediate(); 
  9.   immediate = immediate._idleNext; 
  10.  } 

以上的補償處理就保證了宏任務和微任務的處理能符合預期。

 

責任編輯:姜華 來源: 編程雜技
相關推薦

2021-10-26 06:43:36

NodeJavaScript引擎

2020-10-26 08:34:13

Node.jsCORS前端

2013-11-01 09:34:56

Node.js技術

2015-03-10 10:59:18

Node.js開發指南基礎介紹

2011-09-08 13:46:14

node.js

2011-11-01 10:30:36

Node.js

2011-09-02 14:47:48

Node

2011-09-09 14:23:13

Node.js

2011-11-10 08:55:00

Node.js

2012-10-24 14:56:30

IBMdw

2012-02-03 09:25:39

Node.js

2020-05-29 15:33:28

Node.js框架JavaScript

2022-12-02 23:20:06

Node.jsC++任務管理

2013-04-12 01:51:08

微信公眾平臺接口開發

2023-10-04 07:35:03

2015-06-23 15:27:53

HproseNode.js

2021-04-06 10:15:29

Node.jsHooks前端

2024-07-08 08:53:52

2021-02-01 15:42:45

Node.jsSQL應用程序

2021-09-26 05:06:04

Node.js模塊機制
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 亚洲欧洲成人在线 | 欧美乱操| 韩国av网站在线观看 | 亚洲精品国产第一综合99久久 | 亚洲精品在线看 | 在线免费观看黄色网址 | 久久精品国产久精国产 | 夜夜爽99久久国产综合精品女不卡 | 色婷婷综合久久久中字幕精品久久 | 久久久青草婷婷精品综合日韩 | 精品国产乱码久久久久久丨区2区 | 草久久 | 国产乱码精品一区二区三区忘忧草 | 激情av网站 | 玖玖在线精品 | www.精品国产 | 国产天天操 | 国产精品亚洲一区二区三区在线观看 | 成人福利视频 | 91在线视频观看免费 | 国产三级国产精品 | 在线免费黄色小视频 | 人人干人人干人人干 | 在线观看中文字幕 | 国产在线视频一区二区 | 久久久精品视频一区二区三区 | 久久久影院 | 国产精品免费一区二区三区四区 | 欧美视频第二页 | 搞黄网站在线观看 | 亚洲精品v日韩精品 | 奇米超碰在线 | 色综合色综合色综合 | 亚洲精品www. | 不卡的av在线 | 欧美激情精品久久久久久变态 | 天天干天天草 | 久久久av| 久久天堂网 | h网站在线观看 | 精品一区二区视频 |