JS異步編程中的回調(diào)與promise

最近抽空復(fù)習(xí)了一下之前讀過(guò)的JS書甸各,看了一下關(guān)于回調(diào)函數(shù)和promise相關(guān)部分趣倾。

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

提到異步編程某饰,盡管發(fā)展到如今黔漂,js中解決異步的方式已經(jīng)出現(xiàn)了很多種,Promise牧嫉、async/await... 但不可否認(rèn)酣藻,在這些出現(xiàn)之前辽剧,我們采用的最常規(guī)的方式就是回調(diào)函數(shù),可以說(shuō)偷崩,回調(diào)函數(shù)是js中最基礎(chǔ)的異步模式阐斜。但盡管如此诀紊,回調(diào)還是存在著很多不可忽視的缺點(diǎn)渡紫。

  • 執(zhí)行順序

思考這樣一段代碼

fs.readFile('file.txt', 'utf-8', functino(data){
    console.log('A');
    setTimeout(function () {
        console.log('B')
    }, 0)
    console.log('C')
})
console.log('D'); 

有經(jīng)驗(yàn)的同學(xué)可能稍加推敲就能得出正確結(jié)論惕澎,

D A C B

但不可置否唧喉,這樣一段代碼的執(zhí)行順序是違背我們大腦的正常思維順序的八孝,我們?cè)诖竽X中是不斷上下跳躍著的鸠项。再有,如果把上面的setTimeout換成一個(gè)同步函數(shù)呢楼入?那么結(jié)果就是D A B C嘉熊。再如果它只是會(huì)視情況而定同步或者異步扬舒,也就是我們并不確定它是同步還是異步,這樣的情況下孕惜,我們?nèi)绾谓鉀Q呢诊赊?

解決方法或許只能將每個(gè)步驟硬編碼到前一個(gè)步驟中了。

但是上述只是個(gè)簡(jiǎn)單例子碘箍,現(xiàn)實(shí)中的項(xiàng)目遠(yuǎn)比這個(gè)復(fù)雜鲸郊,嵌套的更深秆撮,狀態(tài)更多,
這種方式使得代碼可復(fù)用性變差盗蟆,維護(hù)成本變高喳资,與我們現(xiàn)在提倡的低耦合相駁腾供。

  • 控制反轉(zhuǎn)
// 假如doSomeThing()是一個(gè)第三方api,負(fù)責(zé)做某些事情
// 通過(guò)傳一個(gè)callback來(lái)執(zhí)行接下來(lái)的步驟
doSomeThing('...', function () {
    // ...
})

上述例子中节值, callback的執(zhí)行取決于doSomeThing()搞疗,這種現(xiàn)象叫做"控制反轉(zhuǎn)"贴汪,如果doSomeThing中發(fā)生異常休吠,或者說(shuō)doSomeThing是一個(gè)你根本不了解的第三方api,那么你所傳的callback可能出現(xiàn)任何你想不到的情況阳懂,因?yàn)榇藭r(shí)callback的控制權(quán)并不在你手中岩调, 你不能決定它何時(shí)調(diào)用,調(diào)用次數(shù)缰揪,是否傳參等等等等....

引用《你不知道的JavaScript中卷》

回調(diào)最大的問(wèn)題是控制反轉(zhuǎn)钝腺,它會(huì)導(dǎo)致信任鏈的完全斷裂艳狐。

總而言之毫目,我們需要一種比回調(diào)更好的機(jī)制,來(lái)解決執(zhí)行順序镀虐、信任的問(wèn)題刮便。值得欣喜的是近零,JS目前已經(jīng)提供了很多更加強(qiáng)大的異步模式久信,Promise就是其中之一裙士。

Promise

所謂 Promise管毙,就是一個(gè)對(duì)象,用來(lái)傳遞異步操作的消息啃炸。它代表了某個(gè)未來(lái)才會(huì)知道結(jié)果的事件(通常是一個(gè)異步操作)卓舵,并且這個(gè)事件提供統(tǒng)一的 API,可供進(jìn)一步處理肿嘲。

Promise 對(duì)象有以下兩個(gè)特點(diǎn)雳窟。

  • 對(duì)象的狀態(tài)不受外界影響匣屡。Promise 對(duì)象代表一個(gè)異步操作耸采,有三種狀態(tài):Pending(進(jìn)行中)虾宇、Resolved(已完成,又稱 Fulfilled)和 Rejected(已失斝癖帷)稀轨。只有異步操作的結(jié)果奋刽,可以決定當(dāng)前是哪一種狀態(tài)艰赞,任何其他操作都無(wú)法改變這個(gè)狀態(tài)方妖。這也是 Promise 這個(gè)名字的由來(lái)党觅,它的英語(yǔ)意思就是「承諾」雌澄,表示其他手段無(wú)法改變镐牺。

  • 一旦狀態(tài)改變任柜,就不會(huì)再變,任何時(shí)候都可以得到這個(gè)結(jié)果摔认。Promise 對(duì)象的狀態(tài)改變宅粥,只有兩種可能:從 Pending 變?yōu)?Resolved 和從 Pending 變?yōu)?Rejected秽梅。只要這兩種情況發(fā)生企垦,狀態(tài)就凝固了钞诡,不會(huì)再變了,會(huì)一直保持這個(gè)結(jié)果接箫。就算改變已經(jīng)發(fā)生了辛友,你再對(duì) Promise 對(duì)象添加回調(diào)函數(shù)废累,也會(huì)立即得到這個(gè)結(jié)果邑滨。這與事件(Event)完全不同驼修,事件的特點(diǎn)是殿遂,如果你錯(cuò)過(guò)了它诈铛,再去監(jiān)聽,是得不到結(jié)果的墨礁。

基本用法

// 我們定義三個(gè)異步行為A幢竹、B、C
function A (cb) {
    setTimeout(function () {
        console.log('執(zhí)行A')
        cb && cb()
    })
}
function B (cb) {
    setTimeout(function () {
        console.log('執(zhí)行B')
        cb && cb()
    })
}
function C (cb) {
    setTimeout(function () {
        console.log('執(zhí)行C')
        cb && cb()
    })
    
}

假設(shè)這三個(gè)行為是相互依賴關(guān)系執(zhí)行恩静,也就是A執(zhí)行完再執(zhí)行B焕毫,B執(zhí)行完再執(zhí)行C
首先看es5的實(shí)現(xiàn)方式

    A(B(C))

在看Promise版本

function A () {
    return new Promise((resolve, reject) => {
        setTimeout(function () {
            console.log('執(zhí)行A')
            resolve()
        })
    })
}
function B () {
    return new Promise((resolve, reject) => {
        setTimeout(function () {
            console.log('執(zhí)行B')
            resolve()
        })
    })
}
function C () {
    return new Promise((resolve, reject) => {
        setTimeout(function () {
            console.log('執(zhí)行C')
            resolve()
        })
    })
}


A().then(B).then(C);

怎么樣蹲坷,是不是覺(jué)得清晰了很多?

回想一下我們?cè)谏厦婊卣{(diào)函數(shù)中遇到的兩個(gè)問(wèn)題 執(zhí)行順序控制反轉(zhuǎn)

  • 執(zhí)行順序

我們可以看到 在promise中我們可以很清晰的看出來(lái)邑飒,先執(zhí)行A接下來(lái)是B然后是C,并且我們也不需要關(guān)心A或者B中是同步還是異步操作循签,無(wú)論同步異步都不會(huì)影響到執(zhí)行順序。
這種方式使得我們的代碼一眼就可以看清楚他的執(zhí)行流程县匠,無(wú)論維護(hù)成本還是清晰程度都比回調(diào)函數(shù)要好的多题山,避免了“Callback Hell(回調(diào)地獄)”

  • 控制反轉(zhuǎn)

Promise擁有個(gè)then方法,then方法的第一個(gè)參數(shù)是resolved狀態(tài)的回調(diào)函數(shù)客峭,第二個(gè)參數(shù)(可選)是rejected狀態(tài)的回調(diào)函數(shù)备蚓。我們可以根據(jù)promise的狀態(tài),如果為resolved,就調(diào)用第一個(gè)回調(diào)函數(shù)况凉,如果狀態(tài)變?yōu)閞ejected烤黍,就調(diào)用第二個(gè)回調(diào)函數(shù)娘赴。這樣我們相當(dāng)于把控制權(quán)重新拿回到我們自己手中。
舉個(gè)例子

function A () {
    return new Promise((resolve, reject) => {
        setTimeout(function () {
            console.log('執(zhí)行A')
            resolve('a')
        })
    })
}

A().then(function(data){
    // data就是A返回的proise狀態(tài)成功后所返回的值
    console.log(data); // 'a'
}, function(err) {
    // 如果A的狀態(tài)變?yōu)閞eject,將會(huì)處罰這個(gè)回調(diào)函數(shù)
})


除了then之外议双,promise還有幾個(gè)方法。

Promise.prototype.catch();

Promise.prototype.catch()方法是.then(null, rejection)或.then(undefined, rejection)的別名,用于指定發(fā)生錯(cuò)誤時(shí)的回調(diào)函數(shù)

promiseFn.then(function(posts) {
  // ...
}).catch(function(error) {
  // 處理 promiseFn 和 前一個(gè)回調(diào)函數(shù)運(yùn)行時(shí)發(fā)生的錯(cuò)誤
  console.log('發(fā)生錯(cuò)誤!', error);
});

Promise.all()

Promise.all()用于將多個(gè) Promise 實(shí)例,包裝成一個(gè)新的 Promise 實(shí)例椅寺。

const p = Promise.all([p1, p2, p3]);

返回的結(jié)果是一個(gè)數(shù)組高镐,里面對(duì)應(yīng)參數(shù)中的幾個(gè)promise實(shí)例的返回值。
只有當(dāng)這幾個(gè)實(shí)例的狀態(tài)都變成成功,或者其中有一個(gè)變?yōu)槭。艜?huì)調(diào)用Promise.all方法后面的回調(diào)函數(shù)。

Promise.race()

Promise.race()方法同樣是將多個(gè) Promise 實(shí)例,包裝成一個(gè)新的 Promise 實(shí)例。

const p = Promise.race([p1, p2, p3]);

但是不同于Promise.all的是,只要p1、p2、p3之中有一個(gè)實(shí)例率先改變狀態(tài),p的狀態(tài)就跟著改變。那個(gè)率先改變的 Promise 實(shí)例的返回值,就傳遞給p的回調(diào)函數(shù)。

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖凉夯,帶你破解...
    沈念sama閱讀 217,406評(píng)論 6 503
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件蹲姐,死亡現(xiàn)場(chǎng)離奇詭異江咳,居然都是意外死亡社露,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,732評(píng)論 3 393
  • 文/潘曉璐 我一進(jìn)店門朝巫,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人,你說(shuō)我怎么就攤上這事肆捕。” “怎么了?”我有些...
    開封第一講書人閱讀 163,711評(píng)論 0 353
  • 文/不壞的土叔 我叫張陵镊辕,是天一觀的道長(zhǎng)删性。 經(jīng)常有香客問(wèn)我,道長(zhǎng),這世上最難降的妖魔是什么译仗? 我笑而不...
    開封第一講書人閱讀 58,380評(píng)論 1 293
  • 正文 為了忘掉前任,我火速辦了婚禮,結(jié)果婚禮上才漆,老公的妹妹穿的比我還像新娘演闭。我一直安慰自己,他們只是感情好侮攀,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,432評(píng)論 6 392
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著帖蔓,像睡著了一般烧栋。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上轴踱,一...
    開封第一講書人閱讀 51,301評(píng)論 1 301
  • 那天,我揣著相機(jī)與錄音桦他,去河邊找鬼。 笑死绸罗,一個(gè)胖子當(dāng)著我的面吹牛菊值,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播柔逼,決...
    沈念sama閱讀 40,145評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼犯助,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼瞬哼!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起人弓,我...
    開封第一講書人閱讀 39,008評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤伸头,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后舷蟀,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體恤磷,經(jīng)...
    沈念sama閱讀 45,443評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,649評(píng)論 3 334
  • 正文 我和宋清朗相戀三年野宜,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了扫步。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 39,795評(píng)論 1 347
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡速缨,死狀恐怖锌妻,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情旬牲,我是刑警寧澤仿粹,帶...
    沈念sama閱讀 35,501評(píng)論 5 345
  • 正文 年R本政府宣布搁吓,位于F島的核電站,受9級(jí)特大地震影響吭历,放射性物質(zhì)發(fā)生泄漏堕仔。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,119評(píng)論 3 328
  • 文/蒙蒙 一晌区、第九天 我趴在偏房一處隱蔽的房頂上張望摩骨。 院中可真熱鬧,春花似錦朗若、人聲如沸恼五。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,731評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)灾馒。三九已至,卻和暖如春遣总,著一層夾襖步出監(jiān)牢的瞬間睬罗,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 32,865評(píng)論 1 269
  • 我被黑心中介騙來(lái)泰國(guó)打工旭斥, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留容达,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 47,899評(píng)論 2 370
  • 正文 我出身青樓垂券,卻偏偏與公主長(zhǎng)得像花盐,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子圆米,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,724評(píng)論 2 354

推薦閱讀更多精彩內(nèi)容

  • 弄懂js異步 講異步之前卒暂,我們必須掌握一個(gè)基礎(chǔ)知識(shí)-event-loop啄栓。 我們知道JavaScript的一大特點(diǎn)...
    DCbryant閱讀 2,710評(píng)論 0 5
  • 前言 編程語(yǔ)言很多的新概念都是為了更好的解決老問(wèn)題而提出來(lái)的娄帖。這篇博客就是一步步分析異步編程解決方案的問(wèn)題以及后續(xù)...
    李向_c52d閱讀 1,068評(píng)論 0 2
  • Promise 是異步編程的一種解決方案。簡(jiǎn)單來(lái)說(shuō) Promise 就是一個(gè)容器昙楚,里面保存著某個(gè)未來(lái)才會(huì)結(jié)束的異步...
    點(diǎn)融黑幫閱讀 784評(píng)論 0 13
  • 你不知道JS:異步 第三章:Promises 在第二章近速,我們指出了采用回調(diào)來(lái)表達(dá)異步和管理并發(fā)時(shí)的兩種主要不足:缺...
    purple_force閱讀 2,067評(píng)論 0 4
  • 參考深入理解 Promise 五部曲 -- 1.異步問(wèn)題[http://www.ghostchina.com/pr...
    合肥黑閱讀 2,296評(píng)論 0 14