摘要:所以,函數(shù)的一個重要實際意義就是用來處理異步操作,改寫回調(diào)函數(shù)。表示在這里等待異步操作返回結果,再繼續(xù)執(zhí)行。
JS異步之Promise,Generator,Async Promise
解決的問題:回調(diào)地獄
Promise規(guī)范:
promise有三種狀態(tài),等待(pending)、已完成(fulfilled/resolved)、已拒絕(rejected).Promise的狀態(tài)只能從“等待”轉到“完成”或者“拒絕”,不能逆向轉換,同時“完成”和“拒絕”也不能相互轉換.
promise 必須提供一個 then方法以訪問其當前值、終值和據(jù)因。promise.then(resolve, reject),resolve 和 reject都是可選參數(shù)。如果 resolve 或reject 不是函數(shù),其必須被忽略.
then 方法必須返回一個 promise 對象.
使用:
實例化promise對象需要傳入函數(shù)(包含兩個參數(shù)),resolve和reject,內(nèi)部確定狀態(tài).resolve和reject函數(shù)可以傳入?yún)?shù)在回調(diào)函數(shù)中使用.
resolve和reject都是函數(shù),傳入的參數(shù)在then的回調(diào)函數(shù)中接收.
var promise = new Promise(function(resolve, reject) { setTimeout(function(){ resolve("好哈哈哈哈"); }); }); promise.then(function(val){ console.log(val) })
then接收兩個函數(shù),分別對應resolve和reject狀態(tài)的回調(diào),函數(shù)中接收實例化時傳入的參數(shù).
promise.then(val=>{ //resolved },reason=>{ //rejected })
catch相當于.then(null, rejection)
當then中沒有傳入rejection時,錯誤會冒泡進入catch函數(shù)中,若傳入了rejection,則錯誤會被rejection捕獲,而且不會進入catch.此外,then中的回調(diào)函數(shù)中發(fā)生的錯誤只會在下一級的then中被捕獲,不會影響該promise的狀態(tài).
new Promise((resolve,reject)=>{ throw new Error("錯誤") }).then(null,(err)=>{ console.log(err,1);//此處捕獲 }).catch((err)=>{ console.log(err,2); }); // 對比 new Promise((resolve,reject)=>{ throw new Error("錯誤") }).then(null,null).catch((err)=>{ console.log(err,2);//此處捕獲 }); // 錯誤示例 new Promise((resolve,reject)=>{ resolve("正常"); }).then((val)=>{ throw new Error("回調(diào)函數(shù)中錯誤") },(err)=>{ console.log(err,1); }).then(null,(err)=>{ console.log(err,2);//此處捕獲,也可用catch });
兩者不等價的情況:
此時,catch捕獲的并不是p1的錯誤,而是p2的錯誤,
p1().then(res=>{ return p2()//p2返回一個promise對象 }).catch(err=> console.log(err))
一個錯誤捕獲的錯誤用例:
該函數(shù)調(diào)用中即使發(fā)生了錯誤依然會進入then中的resolve的回調(diào)函數(shù),因為函數(shù)p1中實例化promise對象時已經(jīng)調(diào)用了catch,若發(fā)生錯誤會進入catch中,此時會返回一個新的promise,因此即使發(fā)生錯誤依然會進入p1函數(shù)的then鏈中的resolve回調(diào)函數(shù).
function p1(val){ return new Promise((resolve,reject)=>{ if(val){ var len = val.length;//傳入null會發(fā)生錯誤,進入catch捕獲錯誤 resolve(len); }else{ reject(); } }).catch((err)=>{ console.log(err) }) }; p1(null).then((len)=>{ console.log(len,"resolved"); },()=>{ console.log("rejected"); }).catch((err)=>{ console.log(err,"catch"); })
Promise回調(diào)鏈:
promise能夠在回調(diào)函數(shù)里面使用 return 和 throw, 所以在then中可以return出一個promise對象或其他值,也可以throw出一個錯誤對象,但如果沒有return,將默認返回 undefined,那么后面的then中的回調(diào)參數(shù)接收到的將是undefined.
function p1(val){ return new Promise((resolve,reject)=>{ val==1?resolve(1):reject() }) }; function p2(val){ return new Promise((resolve,reject)=>{ val==2?resolve(2):reject(); }) }; let promimse = new Promise(function(resolve,reject){ resolve(1) }) .then(function(data1) { return p1(data1)//如果去掉return,則返回undefined而不是p1的返回值,會導致報錯 }) .then(function(data2){ return p2(data2+1) }) .then(res=>console.log(res))Generator函數(shù):
generator函數(shù)使用:
1、分段執(zhí)行,可以暫停
2、可以控制階段和每個階段的返回值
3、可以知道是否執(zhí)行到結尾
function* g() { var o = 1; yield o++; yield o++; } var gen = g(); console.log(gen.next()); // Object {value: 1, done: false} var xxx = g(); console.log(gen.next()); // Object {value: 2, done: false} console.log(xxx.next()); // Object {value: 1, done: false} console.log(gen.next()); // Object {value: undefined, done: true}
generator和異步控制:
利用Generator函數(shù)的暫停執(zhí)行的效果,可以把異步操作寫在yield語句里面,等到調(diào)用next方法時再往后執(zhí)行。這實際上等同于不需要寫回調(diào)函數(shù)了,因為異步操作的后續(xù)操作可以放在yield語句下面,反正要等到調(diào)用next方法時再執(zhí)行。所以,Generator函數(shù)的一個重要實際意義就是用來處理異步操作,改寫回調(diào)函數(shù)。
用法:
async 表示這是一個async函數(shù),await只能用在這個函數(shù)里面。
await 表示在這里等待異步操作返回結果,再繼續(xù)執(zhí)行。
await 后一般是一個promise對象
示例:async用于定義一個異步函數(shù),該函數(shù)返回一個Promise。
如果async函數(shù)返回的是一個同步的值,這個值將被包裝成一個理解resolve的Promise,等同于return Promise.resolve(value)。
await用于一個異步操作之前,表示要“等待”這個異步操作的返回值。await也可以用于一個同步的值。
let timer = async function timer(){ return new Promise((resolve,reject) => { setTimeout(() => { resolve("500"); },500); }); } timer().then(result => { console.log(result); //500 }).catch(err => { console.log(err.message); }); //返回一個同步的值 let sayHi = async function sayHi(){ let hi = await "hello world"; return hi; //等同于return Promise.resolve(hi); } sayHi().then(result => { console.log(result); });
參考:
Promise迷你書
promise筆記
手寫promise
promise實現(xiàn)
文章版權歸作者所有,未經(jīng)允許請勿轉載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉載請注明本文地址:http://specialneedsforspecialkids.com/yun/85043.html
摘要:所以,函數(shù)的一個重要實際意義就是用來處理異步操作,改寫回調(diào)函數(shù)。表示在這里等待異步操作返回結果,再繼續(xù)執(zhí)行。 JS異步之Promise,Generator,Async Promise 解決的問題:回調(diào)地獄 Promise規(guī)范: promise有三種狀態(tài),等待(pending)、已完成(fulfilled/resolved)、已拒絕(rejected).Promise的狀態(tài)只能從...
摘要:所以,函數(shù)的一個重要實際意義就是用來處理異步操作,改寫回調(diào)函數(shù)。表示在這里等待異步操作返回結果,再繼續(xù)執(zhí)行。 JS異步之Promise,Generator,Async Promise 解決的問題:回調(diào)地獄 Promise規(guī)范: promise有三種狀態(tài),等待(pending)、已完成(fulfilled/resolved)、已拒絕(rejected).Promise的狀態(tài)只能從...
摘要:如果你還沒讀過上篇上篇和中篇并無依賴關系,您可以讀過本文之后再閱讀上篇,可戳面試篇寒冬求職季之你必須要懂的原生上小姐姐花了近百個小時才完成這篇文章,篇幅較長,希望大家閱讀時多花點耐心,力求真正的掌握相關知識點。 互聯(lián)網(wǎng)寒冬之際,各大公司都縮減了HC,甚至是采取了裁員措施,在這樣的大環(huán)境之下,想要獲得一份更好的工作,必然需要付出更多的努力。 一年前,也許你搞清楚閉包,this,原型鏈,就能獲得...
摘要:拋出的錯誤對象會被方法回調(diào)函數(shù)接收到命令命令后面是一個對象,返回該對象的結果。有人將其稱之為宏任務微任務,定時器就屬于宏任務的范疇。 前言 上一篇 前端面試題-JavaScript(一), 感興趣的小伙伴也可以移步這里查看 完整版JavaScript面試題,面試題會不定期更新加進去一些個人工作中遇到的或者認為比較重要的東西,后面會涉及到前端的各個方面,感興趣的小伙伴可以關注哦! 如果文...
閱讀 2612·2021-11-16 11:40
閱讀 3409·2021-11-08 13:26
閱讀 871·2021-10-28 09:32
閱讀 3530·2021-09-13 10:26
閱讀 803·2019-08-30 15:55
閱讀 777·2019-08-30 15:44
閱讀 1908·2019-08-30 15:44
閱讀 1756·2019-08-30 13:48