promise

本文是整理阮一峰大神ES6中 Promise 的學習筆記

目錄:

  • Promise.prototype.then()
  • Promise.prototype.catch()
    模擬的是catch代碼塊
  • Promise.prototype.finally()
  • Promise.all()
    全部fulfilled或第一個rejected時回調
  • Promise.race()
    率先改變的 Promise 實例的返回值,就傳遞的回調函數(shù)
  • Promise.allSettled()
    全部結束時回調
  • Promise.any()
    有一個是fulfilled狀態(tài),就是fulfilled狀態(tài),所有都是rejected狀態(tài),才會是rejected狀態(tài)
  • Promise.resolve()
    將現(xiàn)有對象轉為 Promise 對象
  • Promise.reject()
  • Promise.try()
    模擬try代碼塊

Promise對象是一個構造函數(shù),用來生成Promise實例

1.promise狀態(tài)狀態(tài)不受外界影響

  • pending(進行中)
  • fulfilled(已成功)resolved(已定型)
  • rejected(已失?。?/li>

2.狀態(tài)改變,就不會再變

一旦狀態(tài)改變,就不會再變,任何時候都可以得到這個結果。Promise對象的狀態(tài)改變,只有兩種可能:

  • 從pending變?yōu)閒ulfilled
  • 從pending變?yōu)閞ejected

3.Promise缺點

  • 無法取消Promise,一旦新建它就會立即執(zhí)行,無法中途取消。
  • 不設置回調函數(shù),Promise內部拋出的錯誤,不會反應到外部。
  • 當處于pending狀態(tài)時,無法得知目前進展到哪一個階段(剛剛開始還是即將完成)。

4.代碼實例

// 創(chuàng)建promise實例
const promise = new Promise(function(resolve, reject) {
  // 這里的代碼創(chuàng)建后立即執(zhí)行
  // ... some code

  if (/* 異步操作成功 */){
    resolve(value);
  } else {
    reject(error);
  }
});
// promise.then()  接收兩個參數(shù)回調函數(shù),成功函數(shù),失敗函數(shù)
promise.then(function(value) {
  // 全部執(zhí)行完執(zhí)行這里
  // success
}, function(error) {
  // failure
});

5.Promise對象實現(xiàn)的 Ajax 操作

const getJSON = function(url) {
  const promise = new Promise(function(resolve, reject){
    const handler = function() {
      if (this.readyState !== 4) {
        return;
      }
      if (this.status === 200) {
        resolve(this.response);
      } else {
        reject(new Error(this.statusText));
      }
    };
    const client = new XMLHttpRequest();
    client.open("GET", url);
    client.onreadystatechange = handler;
    client.responseType = "json";
    client.setRequestHeader("Accept", "application/json");
    client.send();

  });

  return promise;
};

getJSON("/posts.json").then(function(json) {
  console.log('Contents: ' + json);
}, function(error) {
  console.error('出錯了', error);
});

6.Promise.prototype.then()

then方法是定義在原型對象Promise.prototype上的,返回的是Promise的原型對象的新實例
Promise.prototype ——> new Promise() 原實例
Promise.prototype ——> new Promise().then() 新實例

getJSON("/post/1.json")
.then(function(post) {
  // 這里return的值將傳入下邊的then中
  return getJSON(post.commentURL); 
})
.then(function (comments) {
  // comments這里的參數(shù)就是上邊return的值
  console.log("resolved: ", comments);
}, function (err){
  console.log("rejected: ", err);
});

7.Promise.prototype.catch()

Promise.prototype.catch與下邊等同,但建議用這種方法獲取報錯,因為Promise 對象的錯誤具有“冒泡”性質,最后的catch能獲取到之前n個then中的所有報錯信息

  • .then(null, rejection)
  • .then(undefined, rejection)

異步操作拋出錯誤,狀態(tài)就會變?yōu)閞ejected,就會調用catch方法指定的回調函數(shù),可以寫多個catch和多個then,順序隨意

p.then((val) => console.log('fulfilled:', val))
  .catch((err) => console.log('rejected', err));

// 等同于
p.then((val) => console.log('fulfilled:', val))
  .then(null, (err) => console.log("rejected:", err));

Promise 會吃掉錯誤
Promise 內部的錯誤不會影響到 Promise 外部的代碼

const someAsyncThing = function() {
  return new Promise(function(resolve, reject) {
    // 下面一行會報錯,因為x沒有聲明
    resolve(x + 2);
  });
};

someAsyncThing().then(function() {
  console.log('everything is great');
});

setTimeout(() => { console.log(123) }, 2000);
// Uncaught (in promise) ReferenceError: x is not defined
// 123

8.Promise.prototype.finally()

  • finally方法用于指定不管 Promise 對象最后狀態(tài)如何,都會執(zhí)行的操作。
  • finally方法的回調函數(shù)不接受任何參數(shù)
  • 沒有辦法知道,前面的 Promise 狀態(tài)到底是fulfilled還是rejected。
  • finally方法里面的操作,與狀態(tài)無關的,不依賴于 Promise 的執(zhí)行結果
promise
.finally(() => {
  // 語句
});

// 等同于
promise
.then(
  result => {
    // 語句
    return result;
  },
  error => {
    // 語句
    throw error;
  }
);

9.Promise.all()

Promise.all()方法用于將多個 Promise 實例,包裝成一個新的 Promise 實例

  • Promise.all()方法接受一個數(shù)組作為參數(shù),p1、p2、p3都是 Promise 實例
const p = Promise.all([p1, p2, p3]);

p的狀態(tài)由p1、p2、p3決定,分成兩種情況。

(1)只有p1、p2、p3的狀態(tài)都變成fulfilled,p的狀態(tài)才會變成fulfilled,此時p1、p2、p3的返回值組成一個數(shù)組,傳遞給p的回調函數(shù)。

(2)只要p1、p2、p3之中有一個被rejected,p的狀態(tài)就變成rejected,此時第一個被reject的實例的返回值,會傳遞給p的回調函數(shù)。

// 生成一個Promise對象的數(shù)組
const promises = [2, 3, 5, 7, 11, 13].map(function (id) {
  return getJSON('/post/' + id + ".json");
});

Promise.all(promises).then(function (posts) {
  // ...
}).catch(function(reason){
  // ...
});

promises是包含 6 個 Promise 實例的數(shù)組,只有這 6 個實例的狀態(tài)都變成fulfilled,或者其中有一個變?yōu)閞ejected,才會調用Promise.all方法后面的回調函數(shù)。

const databasePromise = connectDatabase();

const booksPromise = databasePromise
  .then(findAllBooks);

const userPromise = databasePromise
  .then(getCurrentUser);

Promise.all([
  booksPromise,
  userPromise
])
.then(([books, user]) => pickTopRecommendations(books, user));

上面代碼中,booksPromise和userPromise是兩個異步操作,只有等到它們的結果都返回了,才會觸發(fā)pickTopRecommendations這個回調函數(shù)。

注意:如果作為參數(shù)的 Promise 實例,自己定義了catch方法,那么它一旦被rejected,并不會觸發(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('報錯了');
})
.then(result => result)
.catch(e => e);

Promise.all([p1, p2])
.then(result => console.log(result))
.catch(e => console.log(e));
// ["hello", Error: 報錯了]

上面代碼中:
p1會resolved,p2首先會rejected,但是p2有自己的catch方法,該方法返回的是一個新的 Promise 實例,p2指向的實際上是這個實例。該實例執(zhí)行完catch方法后,也會變成resolved,導致Promise.all()方法參數(shù)里面的兩個實例都會resolved,因此會調用then方法指定的回調函數(shù),而不會調用catch方法指定的回調函數(shù)。

10.Promise.race()

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

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

上面代碼中,只要p1、p2、p3之中有一個實例率先改變狀態(tài),p的狀態(tài)就跟著改變。那個率先改變的 Promise 實例的返回值,就傳遞給p的回調函數(shù)。

11.Promise.allSettled()

確保所有操作都結束
Promise.allSettled()方法接受一組 Promise 實例作為參數(shù),包裝成一個新的 Promise 實例。只有等到所有這些參數(shù)實例都返回結果,不管是fulfilled還是rejected,包裝實例才會結束

const promises = [
  fetch('/api-1'),
  fetch('/api-2'),
  fetch('/api-3'),
];

await Promise.allSettled(promises);
removeLoadingIndicator();

上面代碼對服務器發(fā)出三個請求,等到三個請求都結束,不管請求成功還是失敗,加載的滾動圖標就會消失。

12.Promise.any()

只要參數(shù)實例有一個變成fulfilled狀態(tài),包裝實例就會變成fulfilled狀態(tài);如果所有參數(shù)實例都變成rejected狀態(tài),包裝實例就會變成rejected狀態(tài)。

const promises = [
  fetch('/endpoint-a').then(() => 'a'),
  fetch('/endpoint-b').then(() => 'b'),
  fetch('/endpoint-c').then(() => 'c'),
];
try {
  const first = await Promise.any(promises);
  console.log(first);
} catch (error) {
  console.log(error);
}

上面代碼中,Promise.any()方法的參數(shù)數(shù)組包含三個 Promise 操作。其中只要有一個變成fulfilled,Promise.any()返回的 Promise 對象就變成fulfilled。如果所有三個操作都變成rejected,那么就會await命令就會拋出錯誤

13.Promise.resolve()

將現(xiàn)有對象轉為 Promise 對象

Promise.resolve('foo')
// 等價于
new Promise(resolve => resolve('foo'))

參數(shù):

  • 參數(shù)是一個 Promise 實例
    那么Promise.resolve將不做任何修改、原封不動地返回這個實例。
  • 參數(shù)是一個thenable對象
    thenable對象指的是具有then方法的對象,比如下面這個對象。
  • 參數(shù)不是具有then方法的對象,或根本就不是對象
  • 不帶有任何參數(shù)
let thenable = {
  then: function(resolve, reject) {
    resolve(42);
  }
};

let p1 = Promise.resolve(thenable);
p1.then(function(value) {
  console.log(value);  // 42
});

注意: 立即resolve()的 Promise 對象,是在本輪“事件循環(huán)”(event loop)的結束時執(zhí)行,而不是在下一輪“事件循環(huán)”的開始時。

setTimeout(function () {
  console.log('three');
}, 0);

Promise.resolve().then(function () {
  console.log('two');
});

console.log('one');

// one
// two
// three

setTimeout(fn, 0)在下一輪“事件循環(huán)”開始時執(zhí)行
Promise.resolve()在本輪“事件循環(huán)”結束時執(zhí)行
console.log('one')則是立即執(zhí)行,因此最先輸出

14.Promise.reject()

const p = Promise.reject('出錯了');
// 等同于
const p = new Promise((resolve, reject) => reject('出錯了'))

注意,Promise.reject()方法的參數(shù),會原封不動地作為reject的理由,變成后續(xù)方法的參數(shù)。這一點與Promise.resolve方法不一致。

const thenable = {
  then(resolve, reject) {
    reject('出錯了');
  }
};

Promise.reject(thenable)
.catch(e => {
  console.log(e === thenable)
})
// true

15.Promise.try()

事實上,Promise.try就是模擬try代碼塊,就像promise.catch模擬的是catch代碼塊。

const f = () => console.log('now');
Promise.try(f);
console.log('next');
// now
// next
?著作權歸作者所有,轉載或內容合作請聯(lián)系作者
【社區(qū)內容提示】社區(qū)部分內容疑似由AI輔助生成,瀏覽時請結合常識與多方信息審慎甄別。
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發(fā)布,文章內容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務。

相關閱讀更多精彩內容

  • Promise 是異步編程的一種解決方案,比傳統(tǒng)的解決方案 —— 回調函數(shù)和事件 —— 更合理且強大 Promis...
    了凡和纖風閱讀 564評論 0 1
  • async 函數(shù) 含義 ES2017 標準引入了 async 函數(shù),使得異步操作變得更加方便。 async函數(shù)對 ...
    Xyaleo閱讀 1,171評論 0 4
  • 1. Promise 的含義 所謂Promise,簡單說就是一個容器,里面保存著某個未來才會結束的事件(通常是一個...
    ROBIN2015閱讀 580評論 0 0
  • 含義 Promise是異步編程的一種解決方案,用于一個異步操作的最終完成(或失敗)及其結果值的表示,比傳統(tǒng)的回調函...
    nimw閱讀 27,344評論 0 4
  • 獨自回想 那雙明亮的眼睛, 像窗, 連接無窮的天空, 讓我偷偷收藏; 夢里看到 那雙清澈的眼睛, 像溪, 流向未知...
    筆拓的簡書閱讀 440評論 2 4

友情鏈接更多精彩內容