promise

## 前言

今天來分享下promise的用法,es6偉大發明之一,當初我學習的時候也是蠻頭大的,不知道為啥,整個腦子就是,我在哪,我要干啥的懵圈,后面認真學習之后,覺得真是十分好用,下面就來一起學習下吧。

為什么會有promise

首先為什么會有promise的存在,其實很多人都知道的,其中最大的問題就是在處理多個有依賴關系的異步操作時,會出現回調地獄( callback hell ),如下:

$.ajax({url:?'....',success:?function?(data)?{$.ajax({url:?'....',success:?function?(data)?{}});}});

promise提供了一個優雅的方式,來解決這個問題,同時提供了很多的錯誤捕獲機制。

如何使用promise

我們先不講promise的理論語法,這樣會一開始就降低學習的欲望,直接來看使用案例,然后去理解。

首先看基本使用
new Promise(function (resolve, reject) {// 假設此處是異步請求某個數據$.ajax({url:?'......',success:?function?(res)?{if (res.code === 200) {resolve(res.data);} else {reject('獲取data失敗');}}})
})
.then(function A(data) {// 成功,下一步console.log( data);}, function B(error) {// 失敗,做相應處理console.log(error)});
console:
sucess
error

解析:

梳理流程:
  • 首先我們在promise函數里,執行我們的異步操作得到data
  • 如果成功的話,通過resolve函數數據傳遞出來,如果失敗。通過reject把錯誤信息傳遞出來
  • 然后在.then里可以接受傳遞出來的數據,.then()里面接受兩個函數,第一個函數接收resolve傳遞出來的值,也就是正確情況下的處理,第二個函數接收reject傳遞的信息,也就是錯誤的情況下的處理。
Promise是一個對象,它的內部其實有三種狀態。
  • 初始狀態( pending )。
  • 已完成( fulfilled ): Promise 的異步操作已結束成功。
  • 已拒絕( rejected ): Promise 的異步操作未成功結束。

resolve 方法可以使 Promise 對象的狀態改變成成功,同時傳遞一個參數用于后續成功后的操作。
reject 方法則是將 Promise 對象的狀態改變為失敗,同時將錯誤的信息傳遞到后續錯誤處理的操作。

then(onFulfilled, onRejected)

---(onFulfilled, onRejected)

鏈式then

當然,我們既然解決回調地獄,一個異步,看不出來啥優勢,現在看多個異步請求, 為了代碼簡約,我們用setTimeout來代替ajax請求 作為異步操作,如下:

new Promise((resolve, reject) => {setTimeout( () => {if (...){resolve([1, 2, 3])} else {reject('error');}}, 2000);
}).then( data => {console.log(value);  // 打印出[1, 2, 3]return new Promise( (resolve, reject)=> {   // 新的異步請求,需要promise對象let data2 = 1 + data;setTimeout( () => {if (...) {resolve(data2);} else {reject('error2')}}, 2000);});}, error => {cosnole.log(error)})
.then( data2 => {console.log(data2 );}, error => {cosnole.log(error)});
解析:

-這個例子中,第一個異步操作得到數據[1, 2, 3],傳遞到第一個then中,我們在第一個then中運用拿到的數據,進行第二次異步操作,并把結果傳遞出去。在第二個then中拿到數據,并且捕獲error。
可以看到本來嵌套的兩個異步操作,現在清晰多了,而且鏈式接無數個then

在這里有兩個地方需要注意
  • then里面的可捕獲錯誤的函數,可以捕獲到上面的所有then的錯誤,所以只在最后一個then里,寫錯誤捕獲函數就可以。
  • 每次異步操作時候需要返回一個新的promise,因為只有用promise對象才會等異步操作執行完,才去執行下面的then,才能拿到異步執行后的數據,所以第二個then里的異步請求,也需要聲明Promise對象。如果then里面返回常量,可以直接返回。如下:
new Promise((resolve, reject) => {setTimeout( () => {if (...){resolve([1, 2, 3])} else {reject('error');}}, 2000);
}).then( value => {return '222';    // 如果是直接返回常量,可直接return}).then( value2 => {console.log(value2 ); // 打印出222})

下面忽略error情況,看兩個例子,大家可以自己思考下打印結果

new Promise(resolve => {setTimeout( () => {resolve('value1');}, 2000);
}).then( value1 => {console.log(value1);(function () {return new Promise(resolve => {setTimeout(() => {console.log('Mr.Laurence');resolve('Merry Xmas');}, 2000);});}());return false;}).then( value => {console.log(value + ' world');});value1
false world
Mr.Laurence
new Promise( resolve => {console.log('Step 1');setTimeout(() => {resolve(100);}, 1000);
})
.then( value => {return new Promise(resolve => {console.log('Step 1-1');setTimeout(() => {resolve(110);}, 1000);}).then( value => {console.log('Step 1-2');return value;}).then( value => {console.log('Step 1-3');return value;});
})
.then(value => {console.log(value);console.log('Step 2');
});console:
Step 1
Step 1-1
Step 1-2
Step 1-3
110
Step 2

catch

catch 方法是 then(onFulfilled, onRejected) 方法當中 onRejected 函數的一個簡單的寫法,也就是說可以寫成 then(fn).catch(fn),相當于 then(fn).then(null, fn)。使用 catch 的寫法比一般的寫法更加清晰明確。我們在捕獲錯誤的時候,直接在最后寫catch函數即可。

 let promise = new Promise(function(resolve, reject) {throw new Error("Explosion!");
});
promise.catch(function(error) {console.log(error.message); // "Explosion!"
});

上面代碼等于與下面的代碼

 let promise = new Promise(function(resolve, reject) {throw new Error("Explosion!");
});
promise.catch(function(error) {console.log(error.message); // "Explosion!"
});
異步代碼錯誤拋出要用reject
new Promise( resolve => {setTimeout( () => {throw new Error('bye');}, 2000);
})
.then( value => {})
.catch( error => {console.log( 'catch', error);});
控制臺會直接報錯 Uncaught Error: bye

解析:因為異步情況下,catch已經執行完了,錯誤才拋出,所以無法捕獲,所以要用reject,如下:

new Promise( (resolve, reject) => {setTimeout( () => {reject('bye');}, 2000);
})
.then( value => {console.log( value + ' world');})
.catch( error => {console.log( 'catch', error);});catch bye
利用reject可以抓捕到promise里throw的錯
catch 可以捕獲then里丟出來的錯,且按順序只抓捕第一個沒有被捕獲的錯誤
new Promise( resolve => {setTimeout( () => {resolve();}, 2000);
})
.then( value => {throw new Error('bye');})
.then( value => {throw new Error('bye2');})
.catch( error => {console.log( 'catch', error);});console: Error: bye
new Promise( resolve => {setTimeout( () => {resolve();}, 2000);
})
.then( value => {throw new Error('bye');})
.catch( error => {console.log( 'catch', error);})
.then( value => {throw new Error('bye2');})
.catch( error => {console.log( 'catch', error);});console: Error: bye
console: Error: bye2
catch 抓捕到的是第一個沒有被捕獲的錯誤
錯誤被捕獲后,下面代碼可以繼續執行
new Promise(resolve => {setTimeout(() => {resolve();}, 1000);
}).then( () => {throw new Error('test1 error');}).catch( err => {console.log('I catch:', err);   // 此處捕獲了 'test1 error',當錯誤被捕獲后,下面代碼可以繼續執行}).then( () => {console.log(' here');}).then( () => {console.log('and here');throw new Error('test2 error');}).catch( err => {console.log('No, I catch:', err);  // 此處捕獲了 'test2 error'});I catch: Error: test2 error
here
and hereI catch: Error: test2 error
錯誤在捕獲之前的代碼不會執行
new Promise(resolve => {setTimeout(() => {resolve();}, 1000);
}).then( () => {throw new Error('test1 error');}).catch( err => {console.log('I catch:', err);   // 此處捕獲了 'test1 error',不影響下面的代碼執行throw new Error('another error'); // 在catch里面丟出錯誤,會直接跳到下一個能被捕獲的地方。}).then( () => {console.log('and here');throw new Error('test2 error');}).catch( err => {console.log('No, I catch:', err);  // 此處捕獲了 'test2 error'});I catch: Error: test2 error
I catch: Error: another error
new Promise(resolve => {setTimeout(() => {resolve();}, 1000);
}).then( () => {console.log('start');throw new Error('test1 error');}).then( () => {console.log('arrive here');}).then( () => {console.log('... and here');throw new Error('test2 error');  }).catch( err => {console.log('No, I catch:', err);   // 捕獲到了第一個});No, I catch: Error: test1 errorat Promise.then (<anonymous>:8:1

Promise.all

Promise.all([1, 2, 3]).then( all => {console.log('1:', all);})
[1, 2, 3]
Promise.all([function () {console.log('ooxx');}, 'xxoo', false]).then( all => {console.log( all);});[?, "xxoo", false]
let p1 = new Promise( resolve => {setTimeout(() => {resolve('I\'m P1');}, 1500);
});
let p2 = new Promise( (resolve, reject) => {setTimeout(() => {resolve('I\'m P2');}, 1000);});
let p3 = new Promise( (resolve, reject) => {setTimeout(() => {resolve('I\'m P3');}, 3000);});Promise.all([p1, p2, p3]).then( all => {console.log('all', all);
}).catch( err => {console.log('Catch:', err);
});all (3)?["I'm P1", "I'm P2", "I'm P3"]
案例:刪除所有數據后,做一些事情、、、、
db.allDocs({include_docs: true}).then(function (result) {return Promise.all(result.rows.map(function (row) {return db.remove(row.doc);}));
}).then(function (arrayOfResults) {// All docs have really been removed() now!
});

Promise.resolve

Promise.resolve().then( () => {console.log('Step 1');})

其他

Promise.resolve('foo').then(Promise.resolve('bar')).then(function (result) {console.log(result);
});
VM95:2 foo
如果你向 then() 傳遞的并非是一個函數(比如 promise)
它實際上會將其解釋為 then(null),這就會導致前一個 promise 的結果會穿透下面

How do I gain access to resultA here?

function getExample() {return promiseA(…).then(function(resultA) {// Some processingreturn promiseB(…);}).then(function(resultB) {// More processingreturn // How do I gain access to resultA here?});
}

解決 Break the chain

function getExample() {var a = promiseA(…);var b = a.then(function(resultA) {// some processingreturn promiseB(…);});return Promise.all([a, b]).then(function([resultA, resultB]) {// more processingreturn // something using both resultA and resultB});
}

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/449330.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/449330.shtml
英文地址,請注明出處:http://en.pswp.cn/news/449330.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

計算機集群 解說

前些天發現了一個巨牛的人工智能學習網站&#xff0c;通俗易懂&#xff0c;風趣幽默&#xff0c;忍不住分享一下給大家。點擊跳轉到教程。 計算機集群簡稱集群是一種計算機系統&#xff0c;它通過一組松散集成的計算機軟件和/或硬件連接起來高度緊密地協作完成計算工作。 在某…

同時尋找最大數和最小數的最優算法 第二大數

我們知道&#xff0c;在一個容量為n的數據集合中尋找一個最大數&#xff0c;不管用什么樣的比較算法&#xff0c;至少要比較n-1次&#xff0c;就算是用競標賽排序也得比較n-1次&#xff0c;否則你找到的就不能保證是最大的數。那么&#xff0c;在一個容量為n的數據集合中同時尋…

淺談mpvue項目目錄和文件結構

2019獨角獸企業重金招聘Python工程師標準>>> 在Visual Studio Code里面打開項目文件夾&#xff0c;我們可以看到類似如下的文件結構&#xff1a; 1、package.json文件 package.json是項目的主配置文件&#xff0c;里面包含了mpvue項目的基本描述信息、項目所依賴的各…

[AHOI2009]最小割(最大流+tarjan)

繼續填坑了&#xff0c;啦啦啦 這道題本來是準備枚舉每個邊&#xff0c;暫時去除它&#xff0c;但發現時間會爆炸的 于是決定另辟蹊徑 于是這篇題解就應運而生 首先還是網絡流跑一邊 畢竟題目叫最小割嘛&#xff0c;給個面子 然后跑一邊tarjan對滿流的邊處理掉&#xff0c;即不…

進程間通信---信號

什么是信號&#xff1f; 】 信號處理流程 信號類型 發送信號的函數 參數sig&#xff1a;代表 信號 接收信號的函數 參數 handle 的處理方式有幾種&#xff1f; 實例代碼 實例邏輯 圖中的等待操作使用&#xff1a;pause&#xff08;&#xff09;函數 代碼 在這里插入代碼片…

大白話解說,半分鐘就懂 --- 分布式與集群是什么 ? 區別是什么?

前些天發現了一個巨牛的人工智能學習網站&#xff0c;通俗易懂&#xff0c;風趣幽默&#xff0c;忍不住分享一下給大家。點擊跳轉到教程。 PS&#xff1a;這篇文章算是筆記&#xff0c;僅部分文字是原創&#xff0c;相當內容只是收集、整理、提煉、總結別人寫的。 沒有標為原創…

國信證券學習系列(6)

行業輪動策略&#xff1a; 本策略每隔1個月定時觸發計算1000能源&#xff08;399381.SZ&#xff09;、1000材料&#xff08;399382.SZ&#xff09;、1000工業&#xff08;399383.SZ&#xff09;、1000可選&#xff08;399384.SZ&#xff09;、1000消費&#xff08;399385.SZ&a…

用Linux命令行修圖——縮放、編輯、轉換格式——一切皆有可能

本文由 極客范 - 八卦愛好者 翻譯自 How-To Geek。歡迎加入極客翻譯小組&#xff0c;同我們一道翻譯與分享。轉載請參見文章末尾處的要求。ImageMagick是一系列的用于修改、加工圖像的命令行工具。ImageMagick能夠快速地使用命令行對圖片進行操作&#xff0c;對大量的圖片進行…

劍指offer:二維數組中的查找

目錄 題目解題思路具體代碼題目 題目鏈接劍指offer&#xff1a;二維數組中的查找題目描述 在一個二維數組中&#xff08;每個一維數組的長度相同&#xff09;&#xff0c;每一行都按照從左到右遞增的順序排序&#xff0c;每一列都按照從上到下遞增的順序排序。請完成一個函數&a…

函數對象 函數嵌套 名稱空間與作用域

函數對象&#xff1a; 函數是第一類對象&#xff0c;即函數可以當做數據傳遞 1 可以被引用 2 可以當做參數傳遞 3 返回值可以是函數 &#xff08;函數名 不帶&#xff08;&#xff09; 就是函數名的內存地址&#xff0c;帶括號就是執行函數&#xff09; 4 可以當做容器類型的…

國信證券學習系列(7)

跨品種套利策略&#xff1a; 本策略根據計算滾動的.過去的30個bar的均值正負0.5個標準差得到布林線 并在最新價差上穿上軌來做空價差,下穿下軌來做多價差 并在回歸至上下軌水平內的時候平倉 獲取數據&#xff1a; # 獲取兩個品種的收盤價時間序列closesContextInfo.get_ma…

dubbo-admin管理平臺搭建

一、前言 前些天發現了一個巨牛的人工智能學習網站&#xff0c;通俗易懂&#xff0c;風趣幽默&#xff0c;忍不住分享一下給大家。點擊跳轉到教程。 dubbo的使用&#xff0c;其實只需要有注冊中心&#xff0c;消費者&#xff0c;提供者這三個就可以使用了&#xff0c;但是并不能…

不朽傳奇-云計算技術背后的那些天才程序員:Qemu的作者法布里斯貝拉

作者&#xff1a;Liu Guo Hui&#xff0c;OpenStack中國社區&#xff0c;轉載請注明出處 眾所周知&#xff0c;虛擬化技術是構建云基礎架構不可或缺的關鍵技術之一&#xff0c;而在眾多虛擬化技術實現當中&#xff0c;KVM&#xff08;Kernel Virtual Machine&#xff09;因為L…

C學習筆記-字符串

對于C語言來說&#xff0c;字符串其實就是最后一個元素為’\0’的char數組 字符數組的初始化 字符數組常見的有兩種初始化方式 char str[] "hello";或者 char str[] {h, e, l, l, o};當使用sizeof&#xff08;str&#xff09;時&#xff0c;得到的大小為6&#xff…

Shiro安全框架入門篇(登錄驗證實例詳解與源碼)

一、Shiro框架簡單介紹 前些天發現了一個巨牛的人工智能學習網站&#xff0c;通俗易懂&#xff0c;風趣幽默&#xff0c;忍不住分享一下給大家。點擊跳轉到教程。 Apache Shiro是Java的一個安全框架&#xff0c;旨在簡化身份驗證和授權。Shiro在JavaSE和JavaEE項目中都可以使用…

國信證券學習系列(8)

我為什么要用國信&#xff0c;就是這個原因&#xff0c;可以做期權&#xff0c;期貨&#xff0c;股票&#xff0c;etf&#xff0c;可轉債的回測。滿足了我所有的需要&#xff0c;我要做指數增強。通常的做法是股票和期貨。但實際上&#xff0c;股票和期權做組合&#xff0c;成本…

Socket程序從Windows移植到Linux下的一些注意事項

關于這個話題網上流傳的是一個相同的版本&#xff0c;就是那個第一項是頭文件的區別&#xff0c;但后面列出的頭文件只有#include沒有&#xff08;估計是原版的在不斷轉載的過程中有人不小心忘了把尖括號轉義&#xff0c;讓瀏覽器當html標記解析沒了&#xff09;的那個。現在整…

邊緣控制平面Ambassador全解讀

Ambassador是由Datawire開源的一個API網關項目&#xff0c;主要在Kubernetes的容器編排框架中使用。Ambassador本質上是一個通過配置邊緣/API來管理Envoy數據面板的控制面板。而Envoy則是一個基于第7層協議的網絡代理和通信總線&#xff0c;它是一個由Lyft開源的云原生服務&…

Linux 文件編輯命令 詳細整理

前些天發現了一個巨牛的人工智能學習網站&#xff0c;通俗易懂&#xff0c;風趣幽默&#xff0c;忍不住分享一下給大家。點擊跳轉到教程。 一、vi編輯器有3種基本工作模式 首先需要知道vi編輯器有3種基本工作模式&#xff0c;分別是&#xff1a;命令模式、文本輸入模式、和末…

專訪迅雷首席工程師:迅雷的下一代互聯網底層技術構想

摘要&#xff1a;互聯網合縱連橫頻頻上演&#xff0c;迅雷與小米的聯姻也成為了熱點&#xff0c;有許多人為迅雷的上市和迅雷的未來擔憂&#xff0c;這家像工程師一樣的公司&#xff0c;命運會怎樣&#xff0c;他們未來會如何走下去&#xff1f;對此CSDN專訪了迅雷首席工程師劉…