更新時(shí)間:2018-12-26 來(lái)源:黑馬程序員技術(shù)社區(qū) 瀏覽量:
當(dāng)?shù)谝淮螌W(xué)習(xí)編程的時(shí)候,并不知道什么是回調(diào)(Callbacks),那現(xiàn)在同樣假設(shè)你不知道什么是回調(diào)來(lái)進(jìn)行講解,如果我假設(shè)錯(cuò)了,只需要向下滾動(dòng)即可,節(jié)省一下時(shí)間
當(dāng)我第一次學(xué)習(xí)編程時(shí),它幫助我將函數(shù)理解為機(jī)器。這些機(jī)器可以做任何你想要的東西。他們甚至可以接受輸入并返回一個(gè)值。每臺(tái)機(jī)器上都有一個(gè)按鈕,你可以在需要機(jī)器運(yùn)行時(shí)按下該按鈕,即()。
function add (x, y) {
return x + y
}
add(2,3) // 5 - 按下按鈕,執(zhí)行機(jī)器
復(fù)制代碼無(wú)論我按下按鈕,你按下按鈕,或者別人按下按鈕無(wú)所謂。無(wú)論何時(shí)按下按鈕,機(jī)器都將運(yùn)行。
function add (x, y) {
return x + y
}
const me = add
const you = add
const someoneElse = add
me(2,3) // 5 - Press the button, run the machine.
you(2,3) // 5 - Press the button, run the machine.
someoneElse(2,3) // 5 - Press the button, run the machine.
復(fù)制代碼在上面的代碼,我們分配add函數(shù),三個(gè)不同的變量,me,you,和someoneElse。重要的是要注意add我們創(chuàng)建的原始變量和每個(gè)變量都指向內(nèi)存中的相同位置。它們?cè)诓煌拿Q下完全相同。所以,當(dāng)我們調(diào)用me時(shí)you,或者someoneElse,就好像我們正在調(diào)用一樣add函數(shù)。
現(xiàn)在如果我們把a(bǔ)dd機(jī)器送到另一臺(tái)機(jī)器怎么辦?請(qǐng)記住,按下()按鈕并不重要,如果按下它,它就會(huì)運(yùn)行。
function add (x, y) {
return x + y
}
function addFive (x, addReference) {
return addReference(x, 5) // 15 - Press the button, run the machine.
}
addFive(10, add) // 15
復(fù)制代碼你的大腦可能在這一點(diǎn)上有點(diǎn)奇怪,但這里沒有新的東西。我們不是“按下按鈕” add,而是add作為參數(shù)傳遞addFive,重命名它addReference,然后我們“按下按鈕”或調(diào)用它。
這突出了JavaScript語(yǔ)言的一些重要概念。首先,正如你可以將字符串或數(shù)字作為參數(shù)傳遞給函數(shù)一樣,你也可以將函數(shù)的引用作為參數(shù)傳遞。當(dāng)執(zhí)行此操作時(shí),作為參數(shù)傳遞的函數(shù)稱為回調(diào)函數(shù),并且將回調(diào)函數(shù)傳遞給的函數(shù)稱為高階函數(shù)。
因?yàn)樵~匯很重要,所以這里的代碼與重新命名的變量相同,以匹配他們演示的概念。
function add (x,y) {
return x + y
}
function higherOrderFunction (x, callback) {
return callback(x, 5)
}
higherOrderFunction(10, add)
復(fù)制代碼這種模式應(yīng)該看起來(lái)很熟悉,無(wú)處不在。如果你曾經(jīng)使用過(guò)任何JavaScript Array方法,那么你已經(jīng)使用了回調(diào)。如果你曾經(jīng)使用過(guò)lodash,那么你已經(jīng)使用過(guò)回調(diào)。如果你曾經(jīng)使用過(guò)jQuery,那么你已經(jīng)使用了回調(diào)。
[1,2,3].map((i) => i + 5)
_.filter([1,2,3,4], (n) => n % 2 === 0 );
$('#btn').on('click', () =>
console.log('Callbacks are everywhere')
)
復(fù)制代碼通常,回調(diào)有兩種常見的用例。第一,我們看下.map和_.filter例子,是翻轉(zhuǎn)一個(gè)值到另一個(gè)很好的抽象。我們說(shuō)“嘿,這是一個(gè)數(shù)組和一個(gè)函數(shù)。來(lái)吧,根據(jù)我給你的函數(shù)給我一個(gè)新的值“。第二個(gè),也就是我們?cè)趈Query示例中看到的,是將函數(shù)的執(zhí)行延遲到特定時(shí)間?!昂?,這是這個(gè)函數(shù)。每當(dāng)btn點(diǎn)擊具有id的元素時(shí),請(qǐng)繼續(xù)調(diào)用它?!斑@是我們將關(guān)注的第二個(gè)用例,”延遲執(zhí)行函數(shù)直到特定時(shí)間“。
現(xiàn)在我們只看了同步的例子。正如我們?cè)诒疚拈_頭所討論的那樣,我們構(gòu)建的大多數(shù)應(yīng)用程序都沒有預(yù)先獲得所需的所有數(shù)據(jù)。相反,他們需要在用戶與應(yīng)用程序交互時(shí)獲取外部數(shù)據(jù)。我們剛剛看到回調(diào)如何成為一個(gè)很好的用例,因?yàn)樗鼈冊(cè)俅卧试S你“延遲執(zhí)行函數(shù)直到特定時(shí)間”??纯次覀?nèi)绾问乖摼渥舆m應(yīng)數(shù)據(jù)提取并不需要太多想象力。我們可以延遲函數(shù)的執(zhí)行,直到我們獲得所需的數(shù)據(jù),而不是將函數(shù)的執(zhí)行延遲到特定時(shí)間。這可能是最流行的例子,jQuery的方法:getJSON。
// updateUI and showError are irrelevant.
// Pretend they do what they sound like.
const id = 'tylermcginnis'
$.getJSON({
url: `https://api.github.com/users/${id}`,
success: updateUI,
error: showError,
})
復(fù)制代碼在獲得用戶數(shù)據(jù)之前,我們無(wú)法更新應(yīng)用的UI。那么我們?cè)撛趺崔k?我們說(shuō),“嘿,這是一個(gè)對(duì)象。如果請(qǐng)求成功,請(qǐng)繼續(xù)調(diào)用success并傳遞用戶的數(shù)據(jù)。如果沒有,請(qǐng)繼續(xù)調(diào)用error并傳遞錯(cuò)誤對(duì)象。你不需要擔(dān)心每種方法的作用,只要確保在你應(yīng)該的時(shí)候調(diào)用它們。這是使用異步請(qǐng)求回調(diào)的完美演示。
在這一點(diǎn)上,我們已經(jīng)了解了回調(diào)是什么以及它們?nèi)绾卧谕胶彤惒酱a中都有用處的。我們還沒有談到的是回調(diào)的黑暗面。請(qǐng)看下面的代碼。你能說(shuō)出發(fā)生了什么嗎?
// updateUI, showError, and getLocationURL are irrelevant.
// Pretend they do what they sound like.
const id = 'tylermcginnis'
$("#btn").on("click", () => {
$.getJSON({
url: `https://api.github.com/users/${id}`,
success: (user) => {
$.getJSON({
url: getLocationURL(user.location.split(',')),
success (weather) {
updateUI({
user,
weather: weather.query.results
})
},
error: showError,
})
},
error: showError
})
})
復(fù)制代碼如果覺得有幫助,你可以在這里玩實(shí)時(shí)版本。
請(qǐng)注意,我們添加了一些回調(diào)層。首先,我們說(shuō)在btn點(diǎn)擊具有id的元素之前不要運(yùn)行初始的AJAX請(qǐng)求。單擊按鈕后,我們會(huì)發(fā)出第一個(gè)請(qǐng)求。如果該請(qǐng)求成功,我們會(huì)發(fā)出第二個(gè)請(qǐng)求。如果該請(qǐng)求成功,我們將調(diào)用updateUI從兩個(gè)請(qǐng)求獲得的數(shù)據(jù)的方法。無(wú)論你是否乍一看是否理解了代碼,客觀地說(shuō)它比以前的代碼更難閱讀。這將我們帶到“回調(diào)地獄”的主題。
作為人類,我們很自然地會(huì)順序思考。當(dāng)你在嵌套回調(diào)中嵌套回調(diào)時(shí),它會(huì)強(qiáng)迫你超出你自然的思維方式。當(dāng)你的軟件閱讀方式與自然思考方式之間存在脫節(jié)時(shí),就會(huì)發(fā)生錯(cuò)誤。
像大多數(shù)軟件問(wèn)題的解決方案一樣,一種使“回調(diào)地獄”更容易消費(fèi)的常用方法是模塊化你的代碼。
function getUser(id, onSuccess, onFailure) {
$.getJSON({
url: `https://api.github.com/users/${id}`,
success: onSuccess,
error: onFailure
})
}
function getWeather(user, onSuccess, onFailure) {
$.getJSON({
url: getLocationURL(user.location.split(',')),
success: onSuccess,
error: onFailure,
})
}
$("#btn").on("click", () => {
getUser("tylermcginnis", (user) => {
getWeather(user, (weather) => {
updateUI({
user,
weather: weather.query.results
})
}, showError)
}, showError)
})
復(fù)制代碼如果覺得有幫助,你可以在這里玩實(shí)時(shí)版本。
好的,函數(shù)名稱可以幫助我們更加了解正在發(fā)生的事情,但客觀上是“更好”嗎?并不是很多。我們只是在回調(diào)地獄的可讀性問(wèn)題上加了一個(gè)創(chuàng)可貼。問(wèn)題仍然存在,我們自然地按順序思考,即使有額外的功能,嵌套的回調(diào)也會(huì)使我們擺脫順序的思維方式。
下一期回調(diào)與控制反轉(zhuǎn)有關(guān)。當(dāng)你編寫一個(gè)回調(diào)時(shí),假設(shè)你給回調(diào)的程序是負(fù)責(zé)的,并且會(huì)在它應(yīng)該的時(shí)候(并且只有當(dāng)它)時(shí)調(diào)用它。實(shí)際上是將程序控制權(quán)轉(zhuǎn)換為另一個(gè)程序。當(dāng)您處理jQuery,lodash甚至vanilla JavaScript等庫(kù)時(shí),可以安全地假設(shè)使用正確的參數(shù)在正確的時(shí)間調(diào)用回調(diào)函數(shù)。但是,對(duì)于許多第三方庫(kù),回調(diào)函數(shù)是您與它們交互方式的接口。第三方庫(kù)無(wú)論是故意的還是偶然的,都可以打破他們與你的回調(diào)互動(dòng)的方式,這是完全合情合理的。
function criticalFunction () {
// It's critical that this function
// gets called and with the correct
// arguments.
}
thirdPartyLib(criticalFunction)
復(fù)制代碼既然你不是那個(gè)調(diào)用者criticalFunction,你就可以控制調(diào)用它的時(shí)間和參數(shù)。大多數(shù)時(shí)候這不是問(wèn)題,但是當(dāng)它出現(xiàn)問(wèn)題時(shí),這是一個(gè)很大的問(wèn)題。
Promises
你有沒有預(yù)訂去過(guò)一個(gè)繁忙的餐館?當(dāng)這種情況發(fā)生時(shí),餐廳需要一種方法在桌子打開時(shí)與你聯(lián)系。從歷史上看,當(dāng)你的桌子準(zhǔn)備就緒時(shí),他們只會(huì)取你的名字并大喊大叫。然后,自然而然地,他們決定開始變幻想。一個(gè)解決方案是,一旦桌子打開,他們就會(huì)取你的號(hào)碼并給你發(fā)短信,而不是取你的名字。這使您可以超出大喊大叫的范圍,但更重要的是,它允許他們隨時(shí)根據(jù)需要定位你的手機(jī)廣告。聽起來(lái)有點(diǎn)熟?這應(yīng)該!好吧,也許不應(yīng)該。這是回調(diào)的隱喻!將你的號(hào)碼提供給餐館就像給第三方服務(wù)提供回?fù)芄δ芤粯?。你希望餐廳在桌子打開時(shí)給您發(fā)短信,就像你一樣期望第三方服務(wù)在何時(shí)以及如何表達(dá)時(shí)調(diào)用你的功能。一旦你的號(hào)碼或回叫功能掌握在他們手中,您就失去了所有控制權(quán)。
值得慶幸的是,存在另一種解決方案。一個(gè)設(shè)計(jì),允許您保持所有控制。你甚至可能以前都經(jīng)歷過(guò) - 這是他們給你的小嗡嗡聲。你知道,這個(gè)。
如果你之前從未使用過(guò),那么這個(gè)想法很簡(jiǎn)單。他們沒有取你的名字或號(hào)碼,而是給你這個(gè)設(shè)備。當(dāng)設(shè)備開始嗡嗡作響并發(fā)光時(shí),你的桌子就準(zhǔn)備好了。當(dāng)你等待桌子打開時(shí),你仍然可以做任何你想做的事,但現(xiàn)在你不必放棄任何東西。事實(shí)上,恰恰相反。他們必須給你一些東西。沒有控制倒置。
蜂鳴器始終處于三種不同狀態(tài)中的一種- pending,fulfilled或rejected。
pending是默認(rèn)的初始狀態(tài)。當(dāng)他們給你蜂鳴器時(shí),它處于這種狀態(tài)。
fulfilled 當(dāng)蜂鳴器閃爍并且你的桌子準(zhǔn)備就緒時(shí)蜂鳴器所在的狀態(tài)。
rejected當(dāng)出現(xiàn)問(wèn)題時(shí),蜂鳴器處于狀態(tài)。也許餐廳即將關(guān)閉,或者他們忘了有人在晚上出租餐廳。
同樣,要記住的重要一點(diǎn)是,你,蜂鳴器的接收器,擁有所有的控制權(quán)。如果蜂鳴器進(jìn)入fulfilled,你可以去你的桌子。如果它被放入fulfilled并且你想忽略它,那么很酷,你也可以這樣做。如果它被放入rejected,那很糟糕,但你可以去別的地方吃。如果沒有任何事情發(fā)生并且它留在pending,你永遠(yuǎn)不會(huì)吃,但你實(shí)際上并沒有任何東西。
現(xiàn)在你已成為餐廳蜂鳴器的主人,讓我們將這些知識(shí)應(yīng)用到重要的事情上。
如果給餐廳你的號(hào)碼就像給他們一個(gè)回調(diào)功能,接收這個(gè)小小的東西就像收到所謂的“Promise”。
一如既往,讓我們從為什么開始吧。為什么Promises存在?它們的存在使得使異步請(qǐng)求更易于管理的復(fù)雜性。完全像蜂鳴器,一個(gè) Promise可以處于三種狀態(tài)之一pending,fulfilled或者rejected。與蜂鳴器不同,它們代表表示餐館桌子狀態(tài)的這些狀態(tài),它們代表異步請(qǐng)求的狀態(tài)。
如果異步請(qǐng)求仍在進(jìn)行中,則Promise狀態(tài)為pending。如果異步請(qǐng)求成功完成,則Promise狀態(tài)將更改為fulfilled。如果異步請(qǐng)求失敗,Promise則將更改為狀態(tài)rejected。蜂鳴器比喻很有意義,對(duì)嗎?
既然你已經(jīng)理解了Promise存在的原因以及它們可以存在的不同狀態(tài),那么我們還需要回答三個(gè)問(wèn)題。
1、如何創(chuàng)造一個(gè)Promise?
2、如何改變Prommise的狀態(tài)?
3、當(dāng)Promise的狀態(tài)發(fā)生變化時(shí),如何監(jiān)聽?
如何創(chuàng)造一個(gè)Promise?
這個(gè)很直接。創(chuàng)建一個(gè)new實(shí)例Promise。
const promise = new Promise()
復(fù)制代碼如何改變Prommise的狀態(tài)?
該P(yáng)romise構(gòu)造函數(shù)接受一個(gè)參數(shù),一個(gè)(回調(diào))函數(shù)。這個(gè)函數(shù)將傳遞兩個(gè)參數(shù),resolve和reject。
resolve - 一個(gè)允許你更改Promise狀態(tài)的功能 fulfilled
reject- 一個(gè)允許你更改Promise狀態(tài)的功能rejected。
在下面的代碼中,我們使用setTimeout等待2秒然后調(diào)用resolve。這將改變Promise的狀態(tài)fulfilled。
const promise = new Promise((resolve, reject) => {
setTimeout(() => {
resolve() // Change status to 'fulfilled'
}, 2000)
})
復(fù)制代碼我們可以通過(guò)在創(chuàng)建它之后立即記錄promise來(lái)看到這種變化,然后resolve在調(diào)用之后大約2秒后再次記錄。
注意Promise從pending到resolved。
當(dāng)Promise的狀態(tài)發(fā)生變化時(shí),如何監(jiān)聽?
在我看來(lái),這是最重要的問(wèn)題。很酷我們知道如何創(chuàng)建Promise并改變其狀態(tài),但如果我們?cè)跔顟B(tài)發(fā)生變化后不知道如何做任何事情,那就毫無(wú)價(jià)值。
我們還沒有談到的一件事是Promise實(shí)際上是什么。當(dāng)你創(chuàng)建一個(gè)時(shí)new Promise,你真的只是創(chuàng)建一個(gè)普通的舊JavaScript對(duì)象。該對(duì)象可以調(diào)用兩個(gè)方法then,和catch。這是關(guān)鍵。當(dāng)promise的狀態(tài)更改fulfilled為時(shí),.then將調(diào)用傳遞給的函數(shù)。當(dāng)promise的狀態(tài)更改rejected為時(shí),.catch將調(diào)用傳遞給的函數(shù)。這意味著一旦你創(chuàng)建了一個(gè)promise,如果異步請(qǐng)求成功,你將傳遞你想要運(yùn)行的函數(shù).then。如果異步請(qǐng)求失敗,你將傳遞要運(yùn)行的功能.catch。
我們來(lái)看一個(gè)例子吧。我們將setTimeout再次使用fulfilled在兩秒鐘(2000毫秒)之后將Promise的狀態(tài)更改為。
function onSuccess () {
console.log('Success!')
}
function onError () {
console.log('