ES6的Promise

注:此篇文章是我參考阮一峰老師的[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ù)分別是resolvereject。它們是兩個(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))
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末亲族,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子可缚,更是在濱河造成了極大的恐慌霎迫,老刑警劉巖,帶你破解...
    沈念sama閱讀 217,277評(píng)論 6 503
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件帘靡,死亡現(xiàn)場(chǎng)離奇詭異知给,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,689評(píng)論 3 393
  • 文/潘曉璐 我一進(jìn)店門涩赢,熙熙樓的掌柜王于貴愁眉苦臉地迎上來戈次,“玉大人,你說我怎么就攤上這事筒扒∏有埃” “怎么了?”我有些...
    開封第一講書人閱讀 163,624評(píng)論 0 353
  • 文/不壞的土叔 我叫張陵花墩,是天一觀的道長(zhǎng)悬秉。 經(jīng)常有香客問我,道長(zhǎng)冰蘑,這世上最難降的妖魔是什么和泌? 我笑而不...
    開封第一講書人閱讀 58,356評(píng)論 1 293
  • 正文 為了忘掉前任,我火速辦了婚禮祠肥,結(jié)果婚禮上武氓,老公的妹妹穿的比我還像新娘。我一直安慰自己搪柑,他們只是感情好聋丝,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,402評(píng)論 6 392
  • 文/花漫 我一把揭開白布索烹。 她就那樣靜靜地躺著工碾,像睡著了一般。 火紅的嫁衣襯著肌膚如雪百姓。 梳的紋絲不亂的頭發(fā)上渊额,一...
    開封第一講書人閱讀 51,292評(píng)論 1 301
  • 那天,我揣著相機(jī)與錄音垒拢,去河邊找鬼旬迹。 笑死,一個(gè)胖子當(dāng)著我的面吹牛求类,可吹牛的內(nèi)容都是我干的奔垦。 我是一名探鬼主播,決...
    沈念sama閱讀 40,135評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼尸疆,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼椿猎!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起寿弱,我...
    開封第一講書人閱讀 38,992評(píng)論 0 275
  • 序言:老撾萬榮一對(duì)情侶失蹤犯眠,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后症革,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體筐咧,經(jīng)...
    沈念sama閱讀 45,429評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,636評(píng)論 3 334
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了量蕊。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片铺罢。...
    茶點(diǎn)故事閱讀 39,785評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖危融,靈堂內(nèi)的尸體忽然破棺而出畏铆,到底是詐尸還是另有隱情,我是刑警寧澤吉殃,帶...
    沈念sama閱讀 35,492評(píng)論 5 345
  • 正文 年R本政府宣布辞居,位于F島的核電站,受9級(jí)特大地震影響蛋勺,放射性物質(zhì)發(fā)生泄漏瓦灶。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,092評(píng)論 3 328
  • 文/蒙蒙 一抱完、第九天 我趴在偏房一處隱蔽的房頂上張望贼陶。 院中可真熱鬧,春花似錦巧娱、人聲如沸碉怔。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,723評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽撮胧。三九已至,卻和暖如春老翘,著一層夾襖步出監(jiān)牢的瞬間芹啥,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 32,858評(píng)論 1 269
  • 我被黑心中介騙來泰國(guó)打工铺峭, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留墓怀,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 47,891評(píng)論 2 370
  • 正文 我出身青樓卫键,卻偏偏與公主長(zhǎng)得像傀履,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子莉炉,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,713評(píng)論 2 354

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

  • Promiese 簡(jiǎn)單說就是一個(gè)容器钓账,里面保存著某個(gè)未來才會(huì)結(jié)束的事件(通常是一個(gè)異步操作)的結(jié)果,語法上說呢袱,Pr...
    雨飛飛雨閱讀 3,358評(píng)論 0 19
  • Promise的含義: ??Promise是異步編程的一種解決方案官扣,比傳統(tǒng)的解決方案——回調(diào)函數(shù)和事件——更合理和...
    呼呼哥閱讀 2,170評(píng)論 0 16
  • 00惕蹄、前言Promise 是異步編程的一種解決方案,比傳統(tǒng)的解決方案——回調(diào)函數(shù)和事件——更合理和更強(qiáng)大。它由社區(qū)...
    夜幕小草閱讀 2,133評(píng)論 0 12
  • 相信凡是寫過javascript的童鞋也一定都寫過回調(diào)方法(callback),簡(jiǎn)單說回調(diào)方法就是將一個(gè)方法fun...
    ac68882199a1閱讀 86,432評(píng)論 12 90
  • 老師說今天是最后一節(jié)生物實(shí)驗(yàn)課啦泪蔫,突然覺得這些什么煩瑣的對(duì)氨基苯磺酸啊N-1-萘基乙二胺鹽酸之類的真的很好看啊棒旗,有...
    1658d734897e閱讀 130評(píng)論 0 0