同步和異步

JavaScript 的執(zhí)行環(huán)境是單線程的,所謂的單線程就是一次只能完成一個(gè)任務(wù),其任務(wù)的調(diào)度方式就是排隊(duì),這就和火車站洗手間門口的等待一樣,前面的那個(gè)人沒有搞定,你就只能站在后面排隊(duì)等著。

同步和異步

同步:后一個(gè)任務(wù)等待前一個(gè)任務(wù)結(jié)束,然后再執(zhí)行,程序的執(zhí)行順序與任務(wù)的排列順序是一致的、同步的;

異步:每一個(gè)任務(wù)有一個(gè)或多個(gè)回調(diào)函數(shù)(callback),前一個(gè)任務(wù)結(jié)束后,不是執(zhí)行后一個(gè)任務(wù),而是執(zhí)行回調(diào)函數(shù),后一個(gè)任務(wù)則是不等前一個(gè)任務(wù)結(jié)束就執(zhí)行,所以程序的執(zhí)行順序與任務(wù)的排列順序是不一致的、異步的。

在瀏覽器端耗時(shí)很長(zhǎng)的操作都應(yīng)該異步執(zhí)行,避免瀏覽器失去響應(yīng),最好的例子就是Ajax操作。在服務(wù)器端,異步模式甚至是唯一的模式,因?yàn)閳?zhí)行環(huán)境是單線程的,如果允許同步執(zhí)行所有http請(qǐng)求,服務(wù)器性能會(huì)急劇下降,很快就會(huì)失去響應(yīng)。

阻塞與非阻塞

阻塞是指阻塞就是說一個(gè)程序沒運(yùn)行完,它后面的程序是無法運(yùn)行的。

非阻塞是指,一個(gè)程序如果因?yàn)楦鞣N原因(網(wǎng)絡(luò)、代碼量等)沒運(yùn)行完的時(shí)候,其他的程序也是可以繼續(xù)運(yùn)行的。

Javascript 異步編程原理

使用 setTimeout 模擬 JS 中的異步

var foo = function(){ 
    console.log('foo')
    setTimeout(function(){
        console.log('foo 2')
 },1000)}

var bar = function(){
     console.log('bar')
}
foo();
bar();

打印出

foo
bar
foo2

什么樣的函數(shù)為異步的

var xhr = new XMLHttpRequest();
xhr.open('GET','/page.json',true);
xhr.onload = function(){
  cosole.log(xhr.responseText)
}
xhr.send()

在 xhr.open 中我們把第三個(gè)參數(shù)設(shè)置為 true ,也就是異步加載。

常見的異步模型

回調(diào)函數(shù)

這是異步編程最基本的方法。

假定有兩個(gè)函數(shù)f1和f2,后者等待前者的執(zhí)行結(jié)果。

f1();
f2();

把f2寫成f1的回調(diào)函數(shù)

function f1(callback){
  setTimeout(function () {
    // f1的任務(wù)代碼
    callback();
  }, 1000);
}
f1(f2);

回調(diào)函數(shù)的優(yōu)點(diǎn)是簡(jiǎn)單、容易理解和部署,缺點(diǎn)是不利于代碼的閱讀和維護(hù),各個(gè)部分之間高度耦合(Coupling),流程會(huì)很混亂,而且每個(gè)任務(wù)只能指定一個(gè)回調(diào)函數(shù)。

事件監(jiān)聽

f1.on("event", f2);
function f1(){
  setTimeout(function(){
    // f1的任務(wù)代碼
    f1.trigger("event");
  },1000)
}

f1.trigger("event")表示,執(zhí)行完成后,立即觸發(fā) event 事件,從而開始執(zhí)行f2。

JS 和 瀏覽器提供的原生方法基本都是基于事件觸發(fā)機(jī)制的,耦合度很低,不過事件不能得到流程控制。

發(fā)布/訂閱

f2 向"信號(hào)中心" jQuery 訂閱 "done" 信號(hào)。

jQuery.subscribe("done", f2);

f1 進(jìn)行如下改寫:

function f1(){
  setTimeout(function () {
    // f1的任務(wù)代碼
    jQuery.publish("done");
  }, 1000);
}

jQuery.publish("done")的意思是,f1執(zhí)行完成后,向"信號(hào)中心"jQuery發(fā)布"done"信號(hào),從而引發(fā)f2的執(zhí)行。

f2 完成執(zhí)行后,也可以取消訂閱(unsubscribe)。

jQuery.unsubscribe("done", f2);

Promises對(duì)象

Promises對(duì)象是CommonJS工作組提出的一種規(guī)范,目的是為異步編程提供統(tǒng)一接口。

在Promises規(guī)范中,每個(gè)任務(wù)都有三種狀態(tài):默認(rèn)(pending)、完成(fulfilled)、失敗(rejected)。

  • 默認(rèn)狀態(tài)可以單向轉(zhuǎn)移到完成狀態(tài),這個(gè)過程叫resolve,對(duì)應(yīng)的方法是deferred.resolve(promiseOrValue);
  • 默認(rèn)狀態(tài)還可以單向轉(zhuǎn)移到失敗狀態(tài),這個(gè)過程叫reject,對(duì)應(yīng)的方法是deferred.reject(reason);
  • 默認(rèn)狀態(tài)時(shí),還可以通過deferred.notify(update)來宣告任務(wù)執(zhí)行信息,如執(zhí)行進(jìn)度;
  • 狀態(tài)的轉(zhuǎn)移是一次性的,一旦任務(wù)由初始的pending轉(zhuǎn)為其他狀態(tài),就會(huì)進(jìn)入到下一個(gè)任務(wù)的執(zhí)行過程中。
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請(qǐng)結(jié)合常識(shí)與多方信息審慎甄別。
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

  • 同步:一個(gè)任務(wù)A的完成需要依賴另一個(gè)任務(wù)B,只有任務(wù)B完成之后,任務(wù)A才算是完成任務(wù)。異步:一個(gè)任務(wù)A需要依賴任務(wù)...
    _挑燈看劍_閱讀 457評(píng)論 0 2
  • 最近研究Nginx的時(shí)候知道了Nginx工作進(jìn)程是異步非阻塞的方式,就對(duì)同步和異步、阻塞和非阻塞的區(qū)別產(chǎn)生了困惑,...
    Uchiha_Ponny閱讀 553評(píng)論 0 1
  • 作者:嚴(yán)肅鏈接:https://www.zhihu.com/question/19732473/answer/20...
    EmptyBottl_520d閱讀 283評(píng)論 0 0
  • 在Android的API當(dāng)中handler消息處理、Broadcast廣播消息等均采用異步通信機(jī)制。什么是異步通信...
    Karma1026閱讀 5,416評(píng)論 2 8
  • 我在路上想了很多話,現(xiàn)在都忘了,重新拾起來,就成心要疼痛一次。 我在想,這一次,只是作為一個(gè)男人的保鏢去了而已,這...
    lygly9閱讀 256評(píng)論 0 0

友情鏈接更多精彩內(nèi)容