單線程的JavaScript

來源:互聯網
上載者:User

標籤:name   ext   總結   bsp   url   type   單線程   執行   告訴   

Javascript是單線程的

因為JS運行在瀏覽器中,是單線程的,每個window一個JS線程,既然是單線程的,在某個特定的時刻只有特定的代碼能夠被執行,並阻塞其它的代碼。而瀏覽器是事件驅動的(Event driven),瀏覽器中很多行為是非同步(Asynchronized)的,會建立事件並放入執行隊列中。javascript引擎是單線程處理它的任務隊列,你可以理解成就是普通函數和回呼函數構成的隊列。當非同步事件發生時,如mouse click, a timer firing, or an XMLHttpRequest completing(滑鼠點擊事件發生、定時器觸發事件發生、XMLHttpRequest完成回調觸發等),將他們放入執行隊列,等待當前代碼執行完成。

非同步事件驅動

前面已經提到瀏覽器是事件驅動的(Event driven),瀏覽器中很多行為是非同步(Asynchronized)的,例如:滑鼠點擊事件、視窗大小拖拉事件、定時器觸發事件、XMLHttpRequest完成回調等。當一個非同步事件發生的時候,它就進入事件隊列。瀏覽器有一個內部大訊息迴圈,Event Loop(事件迴圈)會輪詢大的事件隊列並處理事件。例如,瀏覽器當前正在忙於處理onclick事件,這時另外一個事件發生了(如:window onSize),這個非同步事件就被放入事件隊列等待處理,只有前面的處理完畢了,空閑了才會執行這個事件。setTimeout也是一樣,當調用的時候,js引擎會啟動定時器timer,大約xxms以後執行xxx,當定時器時間到,就把該事件放到主事件隊列等待處理(瀏覽器不忙的時候才會真正執行)。

瀏覽器不是單線程的

雖然JS運行在瀏覽器中,是單線程的,每個window一個JS線程,但瀏覽器不是單線程的,例如Webkit或是Gecko引擎,都可能有如下線程:

  • javascript引擎線程
  • 介面渲染線程
  • 瀏覽器事件觸發線程
  • Http請求線程

很多童鞋搞不清,如果js是單線程的,那麼誰去輪詢大的Event loop事件隊列?答案是瀏覽器會有單獨的線程去處理這個隊列。

Ajax非同步請求是否真的非同步?

 

很多童鞋搞不清楚,既然說JavaScript是單線程啟動並執行,那麼XMLHttpRequest在串連後是否真的非同步? 
其實請求確實是非同步,這請求是由瀏覽器新開一個線程請求(見前面的瀏覽器多線程)。當請求的狀態變更時,如果先前已設定回調,這非同步線程就產生狀態變更事件放到 JavaScript引擎的事件處理隊列中等待處理。當瀏覽器閒置時候出隊列任務被處理,JavaScript引擎始終是單線程運行回呼函數。javascript引擎確實是單線程處理它的任務隊列,能理解成就是普通函數和回呼函數構成的隊列。

 

總結一下,Ajax請求確實是非同步,這請求是由瀏覽器新開一個線程請求,事件回調的時候是放入Event loop單線程事件隊列等候處理。

 

setTimeout(func, 0)為什麼有時候有用?

 

寫js多的童鞋可能發現,有時候加一個setTimeout(func, 0)非常有用,為什嗎?難道是類比多線程嗎?錯!前面已經說過了,javascript是JS運行在瀏覽器中,是單線程的,每個window一個JS線程,既然是單線程的,setTimeout(func, 0)神奇在哪兒?那就是告訴js引擎,在0ms以後把func放到主事件隊列中,等待當前的代碼執行完畢再執行,注意:重點是改變了代碼流程,把func的執行放到了等待當前的代碼執行完畢再執行。這就是它的神奇之處了。它的用處有三個:

 

  • 讓瀏覽器渲染當前的變化(很多瀏覽器UI render和js執行是放在一個線程中,線程阻塞會導致介面無法更新渲染)
  • 重新評估”script is running too long”警告
  • 改變執行順序

 

非阻塞js的實現(non-blocking javascript)

 

defer

 

<script type="text/javascript" defer src="foo.js"></script>

 

async

<script type="text/javascript" async src="foo.js"></script>

然後第二種方法是動態載入js:

 

setTimeout(function(){    var script = document.createElement("script");    script.type = "text/javascript";    script.src = "foo.js";    var head = true; //加在頭還是尾    if(head)      document.getElementsByTagName("head")[0].appendChild(script);    else      document.body.appendChild(script); }, 0); //另外一個獨立的動態載入js的函數function loadJs(jsurl, head, callback){    var script=document.createElement(‘script‘);    script.setAttribute("type","text/javascript");         if(callback){        if (script.readyState){  //IE            script.onreadystatechange = function(){                if (script.readyState == "loaded" ||                        script.readyState == "complete"){                    script.onreadystatechange = null;                    callback();                }            };        } else {  //Others            script.onload = function(){                callback();            };        }    }    script.setAttribute("src", jsurl);         if(head)     document.getElementsByTagName(‘head‘)[0].appendChild(script);     else      document.body.appendChild(script);  }

 

 

 

單線程的JavaScript

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.