淺談個人對客戶端JavaScript同步、非同步、執行順序等概念的理解

来源:https://www.cnblogs.com/BruceChenAndHisBatCave/archive/2019/10/10/11651228.html
-Advertisement-
Play Games

一.同步和非同步的概念。 同步:即按代碼的順序執行任務。 在下列代碼中,按照同步概念,則是先列印1後列印2。 非同步:即執行一個任務的同時執行另一個任務。如果按照此概念執行上面代碼,則是同時列印出1和2。 二.客戶端JavaScript中代碼的執行順序 首先,不管是核心JavaScript還是客戶端Ja ...


一.同步和非同步的概念。

同步:即按代碼的順序執行任務。

在下列代碼中,按照同步概念,則是先列印1後列印2。

1 console.log(1);
2 console.log(2);

非同步:即執行一個任務的同時執行另一個任務。如果按照此概念執行上面代碼,則是同時列印出1和2。

 

二.客戶端JavaScript中代碼的執行順序

首先,不管是核心JavaScript還是客戶端JavaScript都不包含任何線程機制,只有一個單線程執行模型。單線程即指腳本和事件處理程式在同一時間只能執行一個,不能同時執行,沒有併發性。(HTML5定義了一種為後臺線程的“Web Worker”,本人不甚瞭解,不做贅述)。

單線程的好處在於編程更加簡單,編寫代碼可以確保兩個事件處理程式不會同時運行,操作DOM文檔也不會擔心有其他線程同時修改文檔。但這也意味著JavaScript腳本和事件處理程式不能運行太久,否則會降低網頁的可讀性,甚至導致瀏覽器奔潰的假象。那麼,一個JS程式是怎麼具體執行的呢?

JS的執行任務分為同步任務和非同步任務:

同步任務:指除了非同步任務之外的其他程式。

非同步任務:指各種事件(比如資源載入事件中的loaded、DOMContentLoaded中的回調函數,普通事件中的click,focus,mouseover中的回調函數,window對象的定時器setInterval、setTimeout中的回調函數等等)。

過程:一個js程式執行時,首先將同步任務放入一個執行棧中,先解析同步任務和非同步任務並且按順序執行所有同步任務。當非同步任務被觸發時(如用戶點擊滑鼠或者按下鍵盤),非同步進程處理則會檢測到並將其對應的非同步任務轉移到任務隊列中。同步任務全部執行完畢後,則查看任務隊列中是否有未完成的回調函數,如果有則按順序執行。在此後期間會不斷查看任務隊列並不斷執行,形成事件迴圈。請看如下過程圖

 

三、HTML文件中script標簽的執行順序和其屬性defer、async產生的影響

1.在預設情況下,HTML解析器遇到script標簽時,是先執行腳本,進入腳本並按上面所述的順序執行完代碼。然後再繼續解析渲染HTML頁面文檔,這是對於內聯腳本來說。但同樣的,對於一個由src屬性指定外部文件的腳本來說,也是先下載並執行該腳本。也就是說,在完成改腳本的下載和執行前,其後面的文檔部分都不會顯現出來(實際上DOM樹已經被載入,但是沒被解析為DOM樹)。

以下一個1996最先進的JS代碼可以證明該概念(當時沒有那麼多的非同步事件API實現非同步調用,所以用如下的同步程式來實現動態添加HTML元素)

 1 <!DOCTYPE html>
 2 <html lang="en">
 3 <head>
 4     <meta charset="UTF-8">
 5     <title>Document</title>
 6 </head>
 7 <body>
 8     <h1>Table of Factorials</h1>
 9     <script>
10         function factorial(n) {            //用來實現階乘的函數
11             if(n <= 1) return n;
12             else return n * factorial(n-1);
13         }
14 
15         document.write("<table>");        //開始創建表格        
16         document.write("<tr><th>n</th><th>n!</th></tr>");     //創建表頭        
17         for(var i = 0;i <= 10;i++) {
18             document.write("<tr><td>" + i + "</td><td>" + factorial(i) + "</td></tr>");  //輸出十行表格
19         }
20         document.write("</table>");        //表格結束
21         document.write("Generated at " + new Date());   //輸出時間戳
22     </script>
    <h2>Table of Factorials</h2> 23 </body> 24 </html>

以下為結果圖:

可以得知,腳本的執行在預設的情況下是同步和阻塞的,這是由其單線程模型決定的。但是,對於使用src引入外部文件的script標簽來說,其屬性defer和async可以改變這種情況,實現非同步調用

 

2.對於外聯腳本(即由src屬性引入外部js文件的腳本),其有兩個屬性可以改變同步狀態——deferasync只要有這兩個屬性之一即為非同步腳本

defer(延遲):有了該屬性的外聯腳本會延遲解析執行,即等待文檔的載入和解析完成並可以操作時(不包括img,即可以理解為DOMContentLoaded事件觸發時)才解析執行。看以下代碼:

 1 <!DOCTYPE html>
 2 <html lang="en">
 3 <head>
 4     <meta charset="UTF-8">
 5     <title>Document</title>
 6     <style type="text/css" media="screen">
 7         div {
 8             width: 100px;
 9             height:100px;
10         }
11     </style>
12 </head>
13 <body>
14     <script type="text/javascript" src="console.js" defer></script>
15     <script type="text/javascript">
16         console.log(+new Date());
17     </script>
18     <script type="text/javascript">
19         document.addEventListener("DOMContentLoaded",function(){
20             console.log(+new Date());
21         });
22     </script>
23 </body>
24 </html>

console.js的代碼為:

1 console.log(+new Date());

運行結果截圖:

可見,帶有defer屬性的console.js代碼是第一個內聯js執行後最後一個內聯js執行前才執行的,印證了以上說法

 

async(非同步):HTML解析器在遇到帶有該屬性的腳本時,不會中止頁面文檔的解析,而是一邊下載該腳本一邊繼續後面文檔的解析,一旦腳本下載解析完成則儘快停止文檔解析並回去解析執行該腳本,從而避免了下載腳本時阻塞文檔解析,可以憑此提高文檔解析載入速度。看以下代碼:

 1 <!DOCTYPE html>
 2 <html lang="en">
 3 <head>
 4     <meta charset="UTF-8">
 5     <title>Document</title>
 6     <style type="text/css" media="screen">
 7         div {
 8             width: 100px;
 9             height:100px;
10         }
11     </style>
12 </head>
13 <body>
14     <script type="text/javascript" src="console.js" async></script>
15     <script type="text/javascript">
16         console.log(+new Date());
17     </script>
18     <div>
19         
20     </div>
21 </body>
22 </html>

結果截圖:

可以看出,原本按照預設方式應當先列印的console.js文件反而在內聯script標簽之後執行,可以印證上面所述。

 

那麼,如果兩個屬性都同時擁有呢?這樣的標簽會按照什麼方式執行?答案是瀏覽器會遵從async屬性並忽略defer屬性。

 

註意點:

1.擁有這兩個屬性的script標簽的js文件即為非同步腳本,非同步腳本不能使用document.write()(因為如果用該函數會覆蓋掉其對應標簽解析之前的文檔內容);如下麵代碼:

 1 <!DOCTYPE html>
 2 <html lang="en">
 3 <head>
 4     <meta charset="UTF-8">
 5     <title>Document</title>
 6     <script type="text/javascript" src="console.js" defer></script>
 7     <script type="text/javascript" src="console2.js" async></script>
 8 
 9 </head>
10 <body>
11     
12 </body>
13 </html>

其中console.js和console.log2代碼都為:

1 document.write(1);

結果截圖:

2.defer和async都是布爾屬性,沒有值,只要出現即能激活該屬性

3.defer和async都只適用於外聯腳本,內聯腳本使用這兩個屬性是無效的。

4.defer能訪問完整的文檔樹,無論其腳本位置在何處;而async必定能看到其腳本所在位置之前的文檔樹,但是可能或不可能訪問其後面的文檔內容。

 

四、客戶端JavaScript執行順序的總結

JS程式的執行有兩個階段

第一階段:解析載入HTML文檔的內容,並執行<script>元素里的代碼(包括內聯腳本和外部腳本),通常按其出現順序執行。除非出現defer、async屬性使其成為非同步腳本(詳情見上面defer、async屬性的說明)

第二階段:這個階段是非同步的,而且是由事件驅動的(即有用戶事件才會發生)。在這個階段,一旦用戶產生事件,瀏覽器就會調用之前腳本中的事件處理程式函數,來響應非同步發生的事件(如滑鼠單擊,鍵盤輸入。此時對應的事件處理回調函數被放在了任務隊列中,詳情見第二部分)

 

我們對這兩個階段再進行詳細的劃分,形成一條理想的時間線:

1.Web瀏覽器創建一個Document對象,並開始解析渲染HTML文檔,生成Element對象和Text節點放入文檔中。此時,document.readystate的值為“loading”。

2.當解析HTML文檔過程中遇到沒有async和defer屬性的腳本時,解析器停止解析文檔並開始按順序對腳本進行解析執行,此時腳本內可以便利和操作腳本之前的文檔樹。解析完遇到的腳本後則繼續文檔的解析,以此類推

3.如果遇到帶有async屬性的腳本,瀏覽器會一邊下載該腳本一邊繼續後面文檔內容的解析,當腳本下載解析完畢後立即返回解析執行該腳本。

4.當文檔完成解析時,此時document.readystate的值為interactive

5.然後按照其出現順序繼續解析執行帶有defer屬性的腳本

6.所有的文檔和腳本載入執行渲染完成後(不包括外部載入的圖片多媒體文件等)瀏覽器觸發了Document對象的DOMContentLoaded事件,標志著程式執行從同步腳本執行階段進入到了非同步事件處理事件程式執行階段。註意此時可能還有非同步任務還沒執行完成。

7.此時,文檔已經完全解析完成,但是有一些內容還在載入,如圖片。當這些內容完全載入並且非同步腳本全部載入和執行後,document.readystate的值為“complete”,並且觸發window.onload事件。

8.此刻起,調用非同步事件,以非同步響應用戶輸入事件。

 

註意:這是一條理想的時間線。DOMContentLoaded事件和document.readystate屬性大部分瀏覽器都支持。defer屬性也被大部分瀏覽器支持。而async在IE9及其之前的版本是不支持的。

 


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

-Advertisement-
Play Games
更多相關文章
  • 1. SQL語言包括哪些類型? 數據定義:Create Table,Alter Table,Drop Table, Craete/Drop Index 數據操縱:Select ,insert,update,delete 數據控制:grant,revoke 2. 內聯接,外聯接區別? 內連接是保證兩個 ...
  • 1、如果你使用root用戶進行安裝。 vi /etc/profile 即可 系統變數 2、如果你使用普通用戶進行安裝。 vi ~/.bashrc 用戶變數 export HADOOP_HOME=/export/servers/hadoop-2.8.5 export PATH=$PATH:$HADOO ...
  • 原本是想在酷安上架的,然而審核不通過。。只能通過網頁方式宣傳了 一款使用Jsoup開源庫網路爬蟲的APP,將線上閱讀的小說解析,把小說全本下載為txt文件 由於使用爬蟲技術,所以下載的速度不是很理想,後期可能還得優化優化 下載保存的路徑: 內置sd卡/星之小說下載器 PS:最近吃土了,覺得有用的,捐 ...
  • 試圖更改私有視圖的佈局邊距時出現錯誤 解決方案: ...
  • 如何提高iOS開發技能 1、閱讀博客: "https://github.com/tangqiaoboy/iOSBlogCN" 40多位iOS開發博主的博客地址 2、讀書:每年閱讀一本高質量的iOS開發書籍 3、看WWDC視頻 4、看蘋果的官方文檔 5、看開源項目的代碼 6、多寫代碼,多思考 7、多和 ...
  • 相信很多小伙伴們在項目實戰中,經常會用到界面的 、`載入更多`等功能。需要針對具體功能做針對性開發和調試,耗時耗力。 Paging組件的使用將這部分的工作簡化,從而讓開發者更專註於業務的具體實現。下麵我們一起來學習下Paging組件的使用方法。 首先來看下使用Paging組件實現的分頁載入和刷新效果 ...
  • JSON是一種輕量級的數據格式,用於數據的交互。 Android交互數據主要兩種方式:JSON和 XML。XML格式比JSON格式數量略大,所以大多都使用Json數據格式。 在Android開發的過程中,數據訪問 JSON解析就顯得尤為重要。 示例JSON 添加插件GsonFormat 添加依賴 g ...
  • 函數式編程?利用函數把運算過程封裝起來,通過組合各種函數來計算結果。 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...