国产xxxx99真实实拍_久久不雅视频_高清韩国a级特黄毛片_嗯老师别我我受不了了小说

資訊專欄INFORMATION COLUMN

JS 異步的實現

sihai / 2833人閱讀

摘要:由于引擎同一時間只執行一段代碼這是由單線程的性質決定的,所以每個代碼塊阻塞了其它異步事件的進行。這意味著瀏覽器將等待著一個新的異步事件發生。異步的任務執行的順序是不固定的,主要看返回的速度。

我們經常說JS是單線程的,比如node.js研討會上大家都說JS的特色之一是單線程的,這樣使JS更簡單明了,可是大家真的理解所謂JS的單線程機制嗎?單線程時,基于事件的異步機制又該當如何,這些知識在《JavaScript權威指南》并沒有介紹,我也一直困惑了,直到看到一篇外文,才有了些眉目,這里與大家分享下。后來發現《JavaScript高級程序設計》高級定時器和循環定時器介紹過,不過覺得沒我翻譯這篇原文介紹得更透徹,覺得我寫的不好的,可以查看原外文。

1 先看下兩個例子 1.1. 簡單的settimeout
setTimeout(function () { while (true) { } }, 1000);
setTimeout(function () { alert("end 2"); }, 2000);
setTimeout(function () { alert("end 1"); }, 100);
alert("end");

  執行的結果是彈出’end’、’end 1’,然后瀏覽器假死,就是不彈出‘end 2’。也就是說第一個settimeout里執行的時候是一個死循環,這個直接導致了理論上比它晚一秒執行的第二個settimeout里的函數被阻塞,這個和我們平時所理解的異步函數多線程互不干擾是不符的。

附計時器使用方法

// 初始化一個簡單的js的計時器,一段時間后,才觸發并執行回調函數。 setTimeout 返回一個唯一id,可用這個id來取消這個計時器。
var id = setTimeout(fn,delay);
// 類似于setTimeout,不一樣的是,每隔一段時間,會持續調用回調fn,直到被取消
var id = setInterval(fn,delay);
// 傳入一個計時器的id,取消計時器。
clearInterval(id);
clearTimeout(id);
1.2 ajax請求回調

接著我們來測試一下通過xmlhttprequest實現ajax異步請求調用,主要代碼如下:

var xmlReq = createXMLHTTP();//創建一個xmlhttprequest對象
function testAsynRequest() {
    var url = "/AsyncHandler.ashx?action=ajax";
    xmlReq.open("post", url, true);
    xmlReq.setRequestHeader("Content-Type", "application/x-www-form-urlencoded");
    xmlReq.onreadystatechange = function () {
        if (xmlReq.readyState == 4) {
            if (xmlReq.status == 200) {
                var jsonData = eval("(" + xmlReq.responseText + ")");
                alert(jsonData.message);
            }
            else if (xmlReq.status == 404) {
                alert("Requested URL is not found.");
            } else if (xmlReq.status == 403) {
                alert("Access denied.");
            } else {
                alert("status is " + xmlReq.status);
            }
        }
    };
    xmlReq.send(null);
}
testAsynRequest();//1秒后調用回調函數

while (true) {
}

在服務端實現簡單的輸出:

private void ProcessAjaxRequest(HttpContext context)
{
    string action = context.Request["ajax"];
    Thread.Sleep(1000);//等1秒
    string jsonObject = "{"message":"" + action + ""}";
    context.Response.Write(jsonObject);
}

  理論上,如果ajax異步請求,它的異步回調函數是在多帶帶一個線程中,那么回調函數必然不被其他線程”阻撓“而順利執行,也就是1秒后,它回調執行彈出‘ajax’,可是實際情況并非如此,回調函數無法執行,因為瀏覽器再次因為死循環假死。

  據上面兩個例子,總結如下:

JavaScript引擎是單線程運行的,瀏覽器無論在什么時候都只且只有一個線程在運行JavaScript程序.

JavaScript引擎用單線程運行也是有意義的,單線程不必理會線程同步這些復雜的問題,問題得到簡化。

2. JavaScript引擎

  可JS內部究竟如何實現,我們在接下來探討。

  在了解計時器內部運作前,我們必須清楚一點,觸發和執行并不是同一概念,計時器的回調函數一定會在指定delay的時間后被觸發,但并不一定立即執行,可能需要等待。所有JavaScript代碼是在一個線程里執行的,像鼠標點擊和計時器之類的事件只有在JS單線程空閑時才執行。

  我們來看一下圖表,一開始你可能并沒發現什么或啥都不懂,但請靜下心來,在腦海里繪制出這個場景

  這個圖表中有許多數據信息等著我們去理解,當你完全理解了這個圖,你會對js的異步運行機制(即JavaScript引擎如何實現異步事件)有很好的了解。這個圖是一維的,垂直線上是以毫秒計位,藍色塊代表被劃分的不同的js區域執行代碼。例如,第一個JS區塊執行了18毫秒,鼠標點擊事件被阻塞了將近11毫秒,等等。

  由于JavaScript引擎同一時間只執行一段代碼(這是由JavaScript單線程的性質決定的),所以每個JS代碼塊阻塞了其它異步事件的進行。這意味著當一個異步事件(像鼠標點擊、計時器、Ajax)發生時,這些事件的回調函數將排在隊列后面等待執行(如何排隊完全取決于各瀏覽器,而我們可以忽視它們內部差異,作一個簡化處理)。

  我們首先從第一個JS代碼塊開始,有兩個計時器被初始化:一個10ms的setTimeout和一個10ms的setInterval觀察計時器初始化位置,(計時器初始化完畢后就會開始計時),發現setTimeout計時器的回調實際上會在第一個代碼塊執行完畢前被觸發。但是這里注意的是,它不會立即執行(單線程不能這樣做)。實際上,觸發的回調將被排成一個隊列,等待下一個可執行時間。

  此外,在第一個JS代碼塊,我們發現一個鼠標點擊事件被觸發。這個鼠標點擊JS回調被綁定在異步隊列上(我們從來不知道用戶什么時候執行這個操作,所以它被認為是異步的)且不能馬上執行。像初始化的計時器一樣,排隊等待執行。

  執行完初始化JS代碼塊后,瀏覽器就有個疑問:誰在等待執行?此時,鼠標點擊回調和setTimeout計時器的回調都在等待。瀏覽器將選一個(實際上選擇了“鼠標點擊事件的處理函數”,因為由圖可知它是先進隊的)并立馬執行。而計時器的回調將等待下一合適時機執行。

  注意,鼠標點擊事件執行過程中,interval的回調第一次被觸發,與setTimeout的回調一樣,排隊等待執行。隨著時間推移,等到setTimeout計時器的回調執行時候,setInterval的回調再次被觸發,這次被觸發的回調將被拋棄(因為已經有個同樣的回調在排隊了)。如果一大段代碼塊正在執行,所有的setInterval的回調都將要排隊,一旦大段代碼塊執行完畢,這些一連串的setInterval的回調相互間將被無延遲地執行。實際上,瀏覽器處理setInterval被觸發的回調排隊等待執行時,除非隊列中setInterval回調為空,才允許新的setInterval的回調加入。

  我們發現,setInterval的第一個被觸發的回調執行時,setInterval的回調又被觸發且排到隊列。這向我們傳達一個重要的消息:setInterval不關心目前JS正在執行的內容,setInterval的被觸發的回調都將會無差別地排隊,這意味著兩次setInterval回調函數之間的時間間隔會被犧牲掉(縮減)。

  最后,當setInterval的回調執行兩次后,我們發現沒有javascript引擎要執行東西。這意味著瀏覽器將等待著一個新的異步事件發生。我們知道,在50ms時候,setInterval的回調再次被觸發,但這次并沒有東西阻塞,所以回調就立馬執行了。

  在瀏覽器中,JavaScript引擎是基于事件驅動的,這里的事件可看作是瀏覽器派給它的各種任務,這些任務可能源自當前執行的代碼塊,如調用setTimeout(),也可能來自瀏覽器內核,如onload()、onclick()、onmouseover()、setTimeOut()、setInterval()、Ajax等。如果從代碼的角度來看,所謂的任務實體就是各種回調函數,由于“單線程”的原因,這些任務會進行排隊,一個接著一個等待著被引擎處理。(這段說法來源于PaulGuo)

注意:

一秒一次的setInterver()執行時插入一個耗時10s的setTimeout,可以看到setInterver是繼續執行的,而不是一下蹦出10個setInterver()的回調。

3. 瀏覽器的三個常駐線程

瀏覽器是多線程的,它們在內核制控下相互配合以保持同步。一個瀏覽器至少實現三個常駐線程:JavaScript引擎線程,GUI渲染線程,瀏覽器事件觸發線程(UI線程)。

javascript引擎是基于事件驅動單線程執行的。JS引擎一直等待著event loop中任務的到來,然后加以處理(只有當前函數執行棧執行完畢,才會去任務隊列中取任務執行)。瀏覽器無論什么時候都只有一個JS線程在運行JS程序。

UI渲染線程負責渲染瀏覽器界面,當界面需要重繪(Repaint)或由于某種操作引發回流(reflow)時,該線程就會執行。但是 GUI渲染線程與JS引擎是互斥的,當JS引擎執行時GUI線程會被掛起,JS對頁面的操作即GUI的更新也會被保存在一個隊列中,等到JS引擎空閑時才有機會被執行。這就是JS阻塞頁面加載。

事件觸發線程,當一個事件被觸發時該線程會把事件添加到任務隊列的隊尾,等待JS引擎的處理。這些事件可以來自JavaScript引擎當前執行的代碼塊調用setTimeout/ajax添加一個任務,也可以來自瀏覽器其他線程如鼠標點擊添加的任務。但由于JS的單線程關系,所有這些事件都得排隊等待JS引擎處理。

javascript要等主線程空了才會去查看子線程有沒有回調內容。異步的任務執行的順序是不固定的,主要看返回的速度。

4. JavaScript引擎線程和其它偵聽線程

  瀏覽器內核實現允許多個線程異步執行,這些線程在內核制控下相互配合以保持同步。假如某一瀏覽器內核的實現至少有三個常駐線程: javascript引擎線程,界面渲染線程,瀏覽器事件觸發線程,除些以外,也有一些執行完就終止的線程,如Http請求線程,這些異步線程都會產生不同的異步事件,下面通過一個圖來闡明單線程的JavaScript引擎與另外那些線程是怎樣互動通信的.雖然每個瀏覽器內核實現細節不同,但這其中的調用原理都是大同小異.

上圖中,定時器和事件都按時觸發了,這表明JavaScript引擎的線程和計時器觸發線程、事件觸發線程是三個多帶帶的線程,即使JavaScript引擎的線程被阻塞,其它兩個觸發線程都在運行。

線程間通信:JavaScript引擎執行當前的代碼塊,其它諸如setTimeout給JS引擎添加一個任務,也可來自瀏覽器內核的其它線程,如界面元素鼠標點擊事件,定時觸發器時間到達通知,異步請求狀態變更通知等.從代碼角度看來任務實體就是各種回調函數,JavaScript引擎一直等待著任務隊列中任務的到來.由于單線程關系,這些任務得進行排隊,一個接著一個被引擎處理

GUI渲染也是在引擎線程中執行的(準確描述見下),腳本中執行對界面進行更新操作,如添加結點,刪除結點或改變結點的外觀等更新并不會立即體現出來,這些操作將保存在一個隊列中,待JavaScript引擎空閑時才有機會渲染出來。來看例子(這塊內容還有待驗證,個人覺得當Dom渲染時,才可阻止渲染)

渲染線程與JavaScript引擎線程是互斥的,這容易理解,因為JavaScript腳本是可操縱DOM元素,在修改這些元素屬性同時渲染界面,那么渲染線程前后獲得的元素數據就可能不一致了.見深入理解JavaScript定時機制

test

  這段代碼的本意是從0開始順序顯示數字,它們將一個接一個出現,現在我們來仔細研究一下代碼,while(1)創建了一個無休止的循環,但是對于單線程的JavaScript引擎而言,在實際情況中就會造成瀏覽器暫停響應并處于假死狀態。

  alert()會停止JS引擎的執行,直到按確認鍵,在JS調試的時候,查看當前實時頁面的內容。

5. setTimeout和 setInterval

回到文章開頭,我們來看下setTimeout和setsetInterval的區別。

setTimeout(function(){
    /* Some long block of code ... */
    setTimout(arguments.callee,10);
},10);

setInterval(function(){
    /* Some long block of code ... */
},10);

  這兩個程序段第一眼看上去是一樣的,但并不是這樣。setTimeout代碼至少每隔10ms以上才執行一次;然而setInterval固定每隔10ms將嘗試執行,不管它的回調函數的執行狀態。

我們來總結下:

JavaScript引擎只有一個線程,強制異步事件排隊等待執行。

setTimeout和setInterval在異步執行時,有著根本性不同。

如果一個計時器被阻塞執行,它將會延遲,直到下一個可執行點(這可能比期望的時間更長)

setInterval的回調可能被不停的執行,中間沒間隔(如果回調執行的時間超過預定等待的值)

《JavaScript高級程序設計》中,針對setInterval說法如下:

當使用setInterval()時,僅當沒有該定時器的任何其他代碼實例時,才將定時器代碼添加到隊列中。還要注意兩問題:

某些間隔會被跳過(拋棄);

多個定時器的代碼執行之間的間隔可能會比預期小。此時可采取 setTimeout和setsetInterval的區別 的例子方法。

5. Ajax異步

  很多同學朋友搞不清楚,既然說JavaScript是單線程運行的,那么XMLHttpRequest在連接后是否真的異步?其實請求確實是異步的,不過這請求是由瀏覽器新開一個線程請求(參見上圖),當請求的狀態變更時,如果先前已設置回調,這異步線程就產生狀態變更事件放到JavaScript引擎的處理隊列中等待處理,當任務被處理時,JavaScript引擎始終是單線程運行回調函數,具體點即還是單線程運行onreadystatechange所設置的函數。

  Tip:理解JavaScript引擎運作非常重要,特別是在大量異步事件(連續)發生時,可以提升程序代碼的效率。

網上的帖子大多深淺不一,甚至有些前后矛盾,在下的文章都是學習過程中的總結,如果發現錯誤,歡迎留言指出~

參考:
深入理解JavaScript定時機制
原外文
翻譯參考
部分示例
其它參考1,參考2

PS:歡迎大家關注我的公眾號【前端下午茶】,一起加油吧~

另外可以加入「前端下午茶交流群」微信群,長按識別下面二維碼即可加我好友,備注加群,我拉你入群~

文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規行為,您可以聯系管理員刪除。

轉載請注明本文地址:http://m.specialneedsforspecialkids.com/yun/92020.html

相關文章

  • js學習之異步處理

    摘要:學習開發,無論是前端開發還是都避免不了要接觸異步編程這個問題就和其它大多數以多線程同步為主的編程語言不同的主要設計是單線程異步模型。由于異步編程可以實現非阻塞的調用效果,引入異步編程自然就是順理成章的事情了。 學習js開發,無論是前端開發還是node.js,都避免不了要接觸異步編程這個問題,就和其它大多數以多線程同步為主的編程語言不同,js的主要設計是單線程異步模型。正因為js天生的與...

    VioletJack 評論0 收藏0
  • 異步

    摘要:在異步機制中,任務隊列就是用來維護異步任務回調函數的隊列。四對象對象是工作組提出的一種規范,目的是為異步編程提供統一接口。 異步 1.JavaScript單線程的理解 Javascript語言的執行環境是單線程(single thread)。所謂單線程,就是指一次只能完成一件任務。如果有多個任務,就必須排隊,前面一個任務完成,再執行后面一個任務,以此類推。 2.JavaScript單線...

    goji 評論0 收藏0
  • Javascript異步和回調

    摘要:異步本質上應該就是多線程語言的產物。如果是多線程的異步,假死的應該是運行方法的線程,而方法仍然會按預期打印出。當然了,按我個人的理解,應該說是是的回調函數。 引子 每個故事都有由來。前兩天在看 gulp 的時候,看到了它有個 promise 的玩意兒,然后的然后,這兩天就掉進了 javascript 的異步和回調的坑里面去了。 其間搜索了 javascript promise,看到了...

    CarlBenjamin 評論0 收藏0
  • 由setTimeout深入JavaScript執行環境異步機制

    摘要:圖片轉引自的演講和兩個定時器中回調的執行邏輯便是典型的機制。異步編程關于異步編程我的理解是,在執行環境所提供的異步機制之上,在應用編碼層面上實現整體流程控制的異步風格。 問題背景 在一次開發任務中,需要實現如下一個餅狀圖動畫,基于canvas進行繪圖,但由于對于JS運行環境中異步機制的不了解,所以遇到了一個棘手的問題,始終無法解決,之后在與同事交流之后才恍然大悟。問題的根節在于經典的J...

    codeGoogle 評論0 收藏0
  • 前端面試:js同步與異步問題

    摘要:今天的已經成為一門功能全面的編程語言總結最初的用途是為來實現用戶與瀏覽器的交互二為何是單線程的的單線程,與它的用途有關。這決定了它只能是單線程,否則會帶來很復雜的同步問題。 showImg(https://user-gold-cdn.xitu.io/2019/3/31/169d1c40c27a173c?w=428&h=252&f=png&s=35393); 前言 我本來是打算寫一篇co...

    call_me_R 評論0 收藏0
  • JavaScript 異步

    摘要:從最開始的到封裝后的都在試圖解決異步編程過程中的問題。為了讓編程更美好,我們就需要引入來降低異步編程的復雜性。寫一個符合規范并可配合使用的寫一個符合規范并可配合使用的理解的工作原理采用回調函數來處理異步編程。 JavaScript怎么使用循環代替(異步)遞歸 問題描述 在開發過程中,遇到一個需求:在系統初始化時通過http獲取一個第三方服務器端的列表,第三方服務器提供了一個接口,可通過...

    tuniutech 評論0 收藏0

發表評論

0條評論

sihai

|高級講師

TA的文章

閱讀更多
最新活動
閱讀需要支付1元查看
<