注:此篇文章是我參考阮一峰老師的[ECMAScript 6 入門]文章竣稽,自己記錄的筆記岔擂,詳細(xì)的內(nèi)容請(qǐng)移步阮一峰老師的文章。
1. Promise
var getJSON = function(url) {
var promise = new Promise(function(resolve, reject){
var client = new XMLHttpRequest();
client.open("GET", url);
client.onreadystatechange = handler;
client.responseType = "json";
client.setRequestHeader("Accept", "application/json");
client.send();
function handler() {
if (this.readyState !== 4) {
return;
}
if (this.status === 200) {
resolve(this.response);
} else {
reject(new Error(this.statusText));
}
};
});
return promise;
};
getJSON("/posts.json").then(function(json) {
console.log('Contents: ' + json);
}, function(error) {
console.error('出錯(cuò)了', error);
});
- Promise 是異步編程的一種解決方案。所謂的Promise摩瞎,簡(jiǎn)單說就是一個(gè)容器,里面保存著某個(gè)未來才會(huì)結(jié)束的事件(通常是異步操作)的結(jié)果孝常。
- 從語法來說旗们,Promise是一個(gè)對(duì)象,從它可以獲取異步操作的消息构灸。
- Promise對(duì)象有以下兩個(gè)特點(diǎn)上渴。
(1) 對(duì)象的狀態(tài)受外界影響。Promise對(duì)象代表一個(gè)異步操作喜颁,有三種狀態(tài):Pending(進(jìn)行中)
稠氮、Fulfilled(已成功)
和Rejected(已失敗)
。只有異步操作的結(jié)果半开,可以決定當(dāng)前是哪一種狀態(tài)
(2) 一旦狀態(tài)改變隔披,就不會(huì)再變,任何時(shí)候都可以得到這個(gè)結(jié)果稿茉。Promise對(duì)象的狀態(tài)改變锹锰,只有兩種可能:從Pending
變?yōu)?code>Fulfiled和從Pending
變?yōu)?code>Rejected。只要這兩種情況發(fā)生漓库,狀態(tài)就凝固了恃慧,不會(huì)再變了,會(huì)一直保持這個(gè)結(jié)果渺蒿,這時(shí)就稱為 Resolved(已定型)痢士。如果改變已經(jīng)發(fā)生了,你再對(duì)Promise對(duì)象添加回調(diào)函數(shù)茂装,也會(huì)立即得到這個(gè)結(jié)果怠蹂。這與事件(Event)完全不同,事件的特點(diǎn)是少态,如果你錯(cuò)過了它城侧,再去監(jiān)聽,是得不到結(jié)果的彼妻。 - 基本用法
ES6 規(guī)定嫌佑,Promise對(duì)象是一個(gè)構(gòu)造函數(shù)豆茫,用來生成Promise實(shí)例。
下面代碼創(chuàng)造了一個(gè)Promise實(shí)例屋摇。
var promise = new Promise(function(resolve, reject) {
// ... some code
if (/* 異步操作成功 */){
resolve(value);
} else {
reject(error);
}
});
Promise構(gòu)造函數(shù)接受一個(gè)函數(shù)作為參數(shù)揩魂,該函數(shù)的兩個(gè)參數(shù)分別是resolve
和reject
。它們是兩個(gè)函數(shù)炮温,由 JavaScript 引擎提供火脉,不用自己部署。
resolve
函數(shù)的作用是柒啤,將Promise對(duì)象的狀態(tài)從“未完成”變?yōu)椤俺晒Α保磸?Pending 變?yōu)?Resolved)倦挂,在異步操作成功時(shí)調(diào)用,并將異步操作的結(jié)果担巩,作為參數(shù)傳遞出去妒峦;
reject
函數(shù)的作用是,將Promise對(duì)象的狀態(tài)從“未完成”變?yōu)椤笆 保磸?Pending 變?yōu)?Rejected)兵睛,在異步操作失敗時(shí)調(diào)用肯骇,并將異步操作報(bào)出的錯(cuò)誤,作為參數(shù)傳遞出去祖很。
Promise實(shí)例生成以后笛丙,可以用then方法分別指定Resolved狀態(tài)和Rejected狀態(tài)的回調(diào)函數(shù)。
promise.then(function(value) {
// success
}, function(error) {
// failure
});
then
方法可以接受兩個(gè)回調(diào)函數(shù)作為參數(shù)假颇。第一個(gè)回調(diào)函數(shù)是Promise對(duì)象的狀態(tài)變?yōu)镽esolved時(shí)調(diào)用胚鸯,第二個(gè)回調(diào)函數(shù)是Promise對(duì)象的狀態(tài)變?yōu)镽ejected時(shí)調(diào)用。其中笨鸡,第二個(gè)函數(shù)是可選的姜钳,不一定要提供。這兩個(gè)函數(shù)都接受Promise對(duì)象傳出的值作為參數(shù)形耗。
Promise 新建后就會(huì)立即執(zhí)行哥桥。
let promise = new Promise(function(resolve, reject) {
console.log('Promise');
resolve();
});
promise.then(function() {
console.log('Resolved.');
});
console.log('Hi!');
// Promise
// Hi!
// Resolved
- Promise.prototype.then()
Promise 實(shí)例具有then方法,也就是說激涤,then方法是定義在原型對(duì)象Promise.prototype上的拟糕。它的作用是為 Promise 實(shí)例添加狀態(tài)改變時(shí)的回調(diào)函數(shù)。前面說過倦踢,then方法的第一個(gè)參數(shù)是Resolved狀態(tài)的回調(diào)函數(shù)送滞,第二個(gè)參數(shù)(可選)是Rejected狀態(tài)的回調(diào)函數(shù)。
then方法返回的是一個(gè)新的Promise實(shí)例(注意辱挥,不是原來那個(gè)Promise實(shí)例)犁嗅。因此可以采用鏈?zhǔn)綄懛ǎ磘hen方法后面再調(diào)用另一個(gè)then方法晤碘。
采用鏈?zhǔn)降膖hen褂微,可以指定一組按照次序調(diào)用的回調(diào)函數(shù)奥吩。這時(shí),前一個(gè)回調(diào)函數(shù)蕊梧,有可能返回的還是一個(gè)Promise對(duì)象(即有異步操作),這時(shí)后一個(gè)回調(diào)函數(shù)腮介,就會(huì)等待該P(yáng)romise對(duì)象的狀態(tài)發(fā)生變化肥矢,才會(huì)被調(diào)用。
getJSON("/post/1.json").then(
post => getJSON(post.commentURL)
).then(
comments => console.log("Resolved: ", comments),
err => console.log("Rejected: ", err)
);
上面代碼中叠洗,第一個(gè)then方法指定的回調(diào)函數(shù)甘改,返回的是另一個(gè)Promise對(duì)象。這時(shí)灭抑,第二個(gè)then方法指定的回調(diào)函數(shù)十艾,就會(huì)等待這個(gè)新的Promise對(duì)象狀態(tài)發(fā)生變化。如果變?yōu)镽esolved腾节,就調(diào)用funcA忘嫉,如果狀態(tài)變?yōu)镽ejected,就調(diào)用funcB案腺。
- Promise.prototype.catch()
getJSON('/posts.json').then(function(posts) {
// ...
}).catch(function(error) {
// 處理 getJSON 和 前一個(gè)回調(diào)函數(shù)運(yùn)行時(shí)發(fā)生的錯(cuò)誤
console.log('發(fā)生錯(cuò)誤庆冕!', error);
});
上面代碼中,getJSON方法返回一個(gè) Promise 對(duì)象劈榨,如果該對(duì)象狀態(tài)變?yōu)镽esolved访递,則會(huì)調(diào)用then方法指定的回調(diào)函數(shù);如果異步操作拋出錯(cuò)誤同辣,狀態(tài)就會(huì)變?yōu)镽ejected拷姿,就會(huì)調(diào)用catch方法指定的回調(diào)函數(shù),處理這個(gè)錯(cuò)誤旱函。另外响巢,then方法指定的回調(diào)函數(shù),如果運(yùn)行中拋出錯(cuò)誤棒妨,也會(huì)被catch方法捕獲抵乓。
Promise 對(duì)象的錯(cuò)誤具有“冒泡”性質(zhì),會(huì)一直向后傳遞靶衍,直到被捕獲為止灾炭。也就是說,錯(cuò)誤總是會(huì)被下一個(gè)catch語句捕獲颅眶。
getJSON('/post/1.json').then(function(post) {
return getJSON(post.commentURL);
}).then(function(comments) {
// some code
}).catch(function(error) {
// 處理前面三個(gè)Promise產(chǎn)生的錯(cuò)誤
});
上面代碼中蜈出,一共有三個(gè)Promise對(duì)象:一個(gè)由getJSON產(chǎn)生,兩個(gè)由then產(chǎn)生涛酗。它們之中任何一個(gè)拋出的錯(cuò)誤铡原,都會(huì)被最后一個(gè)catch捕獲偷厦。
一般來說,不要在then方法里面定義Reject狀態(tài)的回調(diào)函數(shù)(即then的第二個(gè)參數(shù))燕刻,總是使用catch方法只泼。
- Promise.all()
方法用于將多個(gè) Promise 實(shí)例,包裝成一個(gè)新的 Promise 實(shí)例卵洗。
var p = Promise.all([p1, p2, p3]);
上面代碼中请唱,Promise.all
方法接受一個(gè)數(shù)組作為參數(shù),p1过蹂、p2十绑、p3
都是 Promise 實(shí)例,如果不是酷勺,就會(huì)先調(diào)用下面講Promise.resolve
方法本橙,將參數(shù)轉(zhuǎn)為 Promise 實(shí)例,再進(jìn)一步處理脆诉。(Promise.all方法的參數(shù)可以不是數(shù)組甚亭,但必須具有 Iterator 接口,且返回的每個(gè)成員都是 Promise 實(shí)例击胜。)
p的狀態(tài)由p1狂鞋、p2、p3決定潜的,分成兩種情況骚揍。
(1) 只有p1、p2啰挪、p3的狀態(tài)都變成fulfilled
信不,p的狀態(tài)才會(huì)變fulfilled
,此時(shí)p1亡呵、p2抽活、p3的返回值組成一個(gè)數(shù)組,傳遞給p的回調(diào)函數(shù)锰什。
(2) 只要p1下硕、p2、p3之中有一個(gè)被rejected
汁胆,p的狀態(tài)就變成rejected
梭姓,此時(shí)第一個(gè)被reject的實(shí)例的返回值,會(huì)傳遞給p的回調(diào)函數(shù)嫩码。
注意誉尖,如果作為參數(shù)的 Promise 實(shí)例,自己定義了catch方法铸题,那么它一旦被rejected铡恕,并不會(huì)觸發(fā)Promise.all()的catch方法琢感。
const p1 = new Promise((resolve, reject) => {
resolve('hello');
})
.then(result => result)
.catch(e => e);
const p2 = new Promise((resolve, reject) => {
throw new Error('報(bào)錯(cuò)了');
})
.then(result => result)
.catch(e => e);
Promise.all([p1, p2])
.then(result => console.log(result))
.catch(e => console.log(e));
// ["hello", Error: 報(bào)錯(cuò)了]
上面代碼中,p1會(huì)resolved探熔,p2首先會(huì)rejected驹针,但是p2有自己的catch方法,該方法返回的是一個(gè)新的 Promise 實(shí)例诀艰,p2指向的實(shí)際上是這個(gè)實(shí)例柬甥。該實(shí)例執(zhí)行完catch方法后,也會(huì)變成resolved涡驮,導(dǎo)致Promise.all()方法參數(shù)里面的兩個(gè)實(shí)例都會(huì)resolved,因此會(huì)調(diào)用then方法指定的回調(diào)函數(shù)喜滨,而不會(huì)調(diào)用catch方法指定的回調(diào)函數(shù)捉捅。
如果p2沒有自己的catch方法,就會(huì)調(diào)用Promise.all()的catch方法虽风。
const p1 = new Promise((resolve, reject) => {
resolve('hello');
})
.then(result => result);
const p2 = new Promise((resolve, reject) => {
throw new Error('報(bào)錯(cuò)了');
})
.then(result => result);
Promise.all([p1, p2])
.then(result => console.log(result))
.catch(e => console.log(e));
// Error: 報(bào)錯(cuò)了
注意 :
catch方法返回一個(gè)新的promise對(duì)象
- Promise.race()
Promise.race方法同樣是將多個(gè)Promise實(shí)例棒口,包裝成一個(gè)新的Promise實(shí)例。
var p = Promise.race([p1, p2, p3]);
上面代碼中辜膝,只要p1无牵、p2、p3之中有一個(gè)實(shí)例率先改變狀態(tài)厂抖,p的狀態(tài)就跟著改變茎毁。那個(gè)率先改變的 Promise 實(shí)例的返回值,就傳遞給p的回調(diào)函數(shù)忱辅。
- Promise.resolve()
有時(shí)需要將現(xiàn)有對(duì)象轉(zhuǎn)為Promise對(duì)象七蜘,Promise.resolve方法就起到這個(gè)作用。
var jsPromise = Promise.resolve($.ajax('/whatever.json'));
上面代碼將jQuery生成的deferred對(duì)象墙懂,轉(zhuǎn)為一個(gè)新的Promise對(duì)象橡卤。
Promise.resolve
等價(jià)于下面的寫法。
Promise.resolve('foo')
// 等價(jià)于
new Promise(resolve => resolve('foo'))
Promise.resolve方法的參數(shù)分成四種情況损搬。
(1) 參數(shù)是一個(gè)Promise實(shí)例
如果參數(shù)是Promise實(shí)例碧库,那么Promise.resolve將不做任何修改、原封不動(dòng)地返回這個(gè)實(shí)例巧勤。
var p = new Promise((resolve,reject)=>{
var a= 10;
resolve(a)
});
p.then(res=>console.log(res));
var p1=Promise.resolve(p);
p1.then(res=>console.log(res))
(2) 參數(shù)是一個(gè)thenable對(duì)象
thenable
對(duì)象指的是具有then方法的對(duì)象嵌灰,比如下面這個(gè)對(duì)象。
let thenable = {
then: function(resolve, reject) {
resolve(42);
}
};
Promise.resolve方法會(huì)將這個(gè)對(duì)象轉(zhuǎn)為Promise對(duì)象颅悉,然后就立即執(zhí)行thenable對(duì)象的then方法伞鲫。
let thenable = {
then: function(resolve, reject) {
resolve(42);
}
};
let p1 = Promise.resolve(thenable);
p1.then(function(value) {
console.log(value); // 42
});
上面代碼中,thenable對(duì)象的then方法執(zhí)行后签舞,對(duì)象p1的狀態(tài)就變?yōu)閞esolved秕脓,從而立即執(zhí)行最后那個(gè)then方法指定的回調(diào)函數(shù)柒瓣,輸出42。
(3) 參數(shù)不是具有then方法的對(duì)象吠架,或根本就不是對(duì)象
如果參數(shù)是一個(gè)原始值芙贫,或者是一個(gè)不具有then方法的對(duì)象,則Promise.resolve方法返回一個(gè)新的Promise對(duì)象傍药,狀態(tài)為Resolved磺平。
var p = Promise.resolve('Hello');
p.then(function (s){
console.log(s)
});
// Hello
上面代碼生成一個(gè)新的Promise對(duì)象的實(shí)例p。由于字符串Hello不屬于異步操作(判斷方法是字符串對(duì)象不具有then方法)拐辽,返回Promise實(shí)例的狀態(tài)從一生成就是 Resolved拣挪,所以回調(diào)函數(shù)會(huì)立即執(zhí)行。Promise.resolve方法的參數(shù)俱诸,會(huì)同時(shí)傳給回調(diào)函數(shù)菠劝。
(4) 不帶有任何參數(shù)
Promise.resolve
方法允許調(diào)用時(shí)不帶參數(shù),直接返回一個(gè)Resolved狀態(tài)的Promise對(duì)象睁搭。
所以赶诊,如果希望得到一個(gè)Promise對(duì)象,比較方便的方法就是直接調(diào)用Promise.resolve方法园骆。
var p = Promise.resolve();
p.then(function () {
// ...
});
上面代碼的變量p就是一個(gè)Promise對(duì)象舔痪。
需要注意的是,立即resolve的Promise對(duì)象锌唾,是在本輪“事件循環(huán)”(event loop)的結(jié)束時(shí)锄码,而不是在下一輪“事件循環(huán)”的開始時(shí)。
setTimeout(function () {
console.log('three');
}, 0);
Promise.resolve().then(function () {
console.log('two');
});
console.log('one');
// one
// two
// three
上面代碼中晌涕,setTimeout(fn, 0)在下一輪“事件循環(huán)”開始時(shí)執(zhí)行巍耗,Promise.resolve()在本輪“事件循環(huán)”結(jié)束時(shí)執(zhí)行,console.log('one')則是立即執(zhí)行渐排,因此最先輸出炬太。
- Promise.reject(value)
Promise.reject(reason)方法也會(huì)返回一個(gè)新的 Promise 實(shí)例,該實(shí)例的狀態(tài)為rejected驯耻。
var p4 =Promise.reject('Mistake');
p4.catch(err => console.log(err))