[Effective JavaScript 筆記]第61條:不要阻塞I/O事件隊列

来源:http://www.cnblogs.com/wengxuesong/archive/2016/07/15/5674544.html
-Advertisement-
Play Games

js程式是構建在事件之上的。輸入可能來自不同的外部源。在一些語言中,我們習慣地編寫代碼來等待某個特定的輸入。var text=downloadSync('http://example.com/file.txt'); console.log(text); 像這樣的形式downloadSync稱為同步函... ...


js程式是構建在事件之上的。輸入可能來自不同的外部源。在一些語言中,我們習慣地編寫代碼來等待某個特定的輸入。

var text=downloadSync('http://example.com/file.txt');
console.log(text);

像這樣的形式downloadSync稱為同步函數(或阻塞函數)。程式會停止做任何工作,而等待它的輸入。在這個例子中,也就是等待從網路上下載文件的結果。由於在等待下載完成的期間,電腦可以做其他有用的工作,因此這樣的語言通常為程式員提供一種方法來創建多個線程,即並行執行子計算。它允許程式的一部分停下來等待(阻塞)一個低速的輸入,而程式的另一部分可以繼續進行獨立的工作。
在js中,大多的I/O操作都提供了非同步的或非阻塞的API。程式員提供一個回調函數,一旦輸入完成就可以被系統調用,而不是程式阻塞在等待結果的線程上。

var text=downloadSync('http://example.com/file.txt',function(text){
  console.log(text);
});

該API初始化下載進行,然後在內部註冊表中存儲了回調函數後立即返回,而不是被網路請求阻塞。當下載完成之後,系統會將下載完的文件的文本作為參數調用該已註冊的回調函數。
隨著事件的發生,事件被添加到應用程式的事件隊列的末尾。js系統使用一個內部迴圈機制來執行應用程式。該迴圈機制每次都拉取隊列底部的事件,以接收到這些事件的順序來調用這些已經註冊的js事件處理程式,並將事件的數據作為該事件處理程式的參數。
運行到完成機制擔保的好處是當代碼運行時,你完全掌握應用程式的狀態。根本不必擔心一些變數和對象屬性的改變由於併發執行代碼而超出你的控制。併發編程在js中往往比使用線程和鎖的c++,java或c#更容易。
然而,運行到完成機制的不足是,實際上所有你編寫的代碼支撐著餘下應用程式的繼續執行。像瀏覽器這樣的互動式應用程式中,一個阻塞的事件處理程式會阻塞任何將被處理的其他用戶輸入,甚至可能阻塞一個頁面的渲染,從而導致頁面失去響應的用戶體驗。在伺服器環境中,一個阻塞的事件處理程式可能會阻塞將被處理的其他網路請求,從而導致伺服器失去響應。
js併發的一個最重要的規則是絕不要在應用程式事件隊列中使用阻塞I/O的API。在瀏覽器中,甚至幾乎沒有任何阻塞API是可用的,儘管有一些平臺實現了。提供類似於downloadAsync功能的網路I/O的XMLHttpRequest庫有一個同步的版本實現,被認為是不好的。對於web應用程式的交互性,同步的I/O會導致災難性的後果,它在I/O操作完成之前一直會阻塞用戶與頁面的交互。
相比之下,非同步的API用在基於事件的環境中是安全的,它們迫使應用程式邏輯在一個獨立的事件迴圈“輪詢”中繼續處理。如上面的例子,假設需要幾秒鐘來下載網路資源,在這段時間里,數量龐大的其他事件很可能發生。在同步的實現中,這些事件就會堆積在事件隊列中,而事件迴圈將停留等待該JS代碼執行完成,這將阻塞任何其他事件的處理。在非同步的版本中,JS代碼註冊一個事件處理程式並立即返回,這將在下載完成之前,允許其他處理程式處理這期間的事件。
在主應用程式事件隊列不受影響的環境中,阻塞操作很少出問題。例如,web平臺提供了Worker的API,該API使得產生大量的並行計算成為可能。不同於傳統的線程執行,Workers在一個完全隔離的狀態下執行,沒有獲取全局作用域或應用程式主線程web頁面內容的能力。因此,它們不會妨礙主事件隊列中運行的代碼的執行。在一個Worker中,使用XMLHttpRequest同步的變種很少出問題。下載操作的確會阻塞Worker繼續執行,但這並不會阻止頁面的渲染或事件隊列中的事件響應。在伺服器端環境中,阻塞的API在啟動一開始是沒有問題的,也就是在伺服器開始響應輸入的請求之前。然後在處理請求期間,瀏覽器事件隊列中存在阻塞的API就是有問題的啦。

提示

  • 非同步API使用回調函數來延緩處理代價高昂的操作以避免阻塞主應用程式

  • js併發地接收事件,但會使用一個事件隊列按序地處理事件處理程式

  • 在應用程式事件隊列中絕不要使用阻塞的I/O


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 一、基於 XML 的 Bean 的配置 1.通過屬性註入 即通過 setXxx() 方法註入 Bean 的屬性值或依賴的對象。 使用 <property name="" value="" ref=""/> 元素,其中 name 值為對應 Bean 屬性名稱,value 指定對應屬性值,ref 引用其 ...
  • 一、學習版本 spring-framework-4.0.0 二、導入 jar 包: 三、在類路徑下創建 Spring Config 文件:ApplicationContext.xml 四、創建一個非侵入的 Bean 五、在 Spring Config 文件中配置該 Bean 六、通過 IOC 容器對 ...
  • 字元串:是以空字元\0為結尾的char數組 在程式中定義字元串的方法 1.字元串常量(字元串文字):是指位於一對雙引號中的任何字元,雙引號里的字元加上編譯器自動提供的結束標誌\0字元,作為一個字元串背儲存在記憶體里。它常常用作printf和puts的參數。 2.字元串數組:定義一個字元串數組時,必須讓 ...
  • 1. 在 Form 上放一個 TImage ,再一個 TText 到 Image 裡面,並將 Image1.StyleName 設定為 BtnStyle,如下: 2.接著放一個 TButton,將 StyleLookup 指定為 BtnStyle,按鈕就變成這個 Style,如下: ...
  • 本文為原創文章,轉載請註明出處,謝謝 Curator使用 1、jar包引入,演示版本為2.6.0,非maven項目,可以下載jar包導入到項目中 2、RetryPolicy:重試機制 ExponentialBackoffRetry:每次重試會增加重試時間baseSleepTimeMs Exponen ...
  • × 目錄 [1]原始值 [2]對象 [3]顯式[4]隱式 前面的話 所有程式設計語言的重要特征是具有進行類型轉換的能力,javascript給開發者提供了大量簡單的類型轉換方法。javascript是一門弱類型語言,所以類型轉換成為其比較複雜的一部分。本文將從原始值轉換成原始值、對象轉換成原始值、顯 ...
  • 1.本次用到錯誤提示文字的顏色 http://v3.bootcss.com/css/#forms jquery 教程: http://www.w3school.com.cn/jquery/index.asp 2.工程地址:https://github.com/digitalClass/web_pag ...
  • 這一部分我們來簡單應用一下line-height屬性。 一、單行文本垂直居中 代碼如下: 頁面效果: 二、圖像水平垂直居中 代碼如下: 頁面效果: 其實現原理,我們可以有兩種方式解讀。 第一種方式,先讓隱藏文本節點垂直居中,然後圖像與它垂直居中對齊。 來看這段代碼: 其頁面效果為: 前面我們提到過, ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...