Promise详解
promise 学习笔记
一、准备部分
1)函数对象与实例对象
-
函数对象:将函数作为对象使用时 成为函数对象:
$.get('/test')
,利用jQuery发出Ajax请求,左边是点,就是当作对象用;
-
实例对象:new 函数产生的对象,简称对象:
function Fn() {}; const fn = new Fn()
,Fn是构造函数,fn是实例对象。
2)两种类型的回调函数
回调函数一般包含以下三种定义:1) 自己定义的 不是系统具备的;2) 自己一般不会去亲自调用;3) 最后都被执行了,一般是在某种情况下被自动调用了(如setTimeOut的回调函数)。
- 同步回调:
- 立即执行,完全执行完了才结束,不会放入回调队列;
- 例如:数组遍历相关的回调函数(forEach) / Promise的excutor函数;
const arr = [1, 2, 3];
arr.forEach(item => {
console.log(item);
})
console.log('after forEach');
//这里会等上面数组遍历完毕,才打印下面的字符串
- 异步回调:
- 不会立即执行,会放入回调队列中将来执行;
- 例如:定时器回调 / Ajax回调 / Promise的resolved rejected 的回调;
var timer = setTimeout(() => {
console.log('setTimeout callback()')
})
console.log('after setTimeout');
//先下再上
3)错误
a. 错误类型
-
EvalError
创建一个error实例,表示错误的原因:与 eval()有关。
-
InternalError
创建一个代表Javascript引擎内部错误的异常抛出的实例。 如: "递归太多".
-
RangeError
创建一个error实例,表示错误的原因:数值变量或参数超出其有效范围吗,或者递归层数过多。
-
ReferenceError
创建一个error实例,表示错误的原因:无效引用。
-
SyntaxError
创建一个error实例,表示错误的原因:eval()在解析代码的过程中发生的语法错误。
-
TypeError
创建一个error实例,表示错误的原因:变量或参数不属于有效类型。
-
URIError
创建一个error实例,表示错误的原因:给 encodeURI()或 decodeURI()传递的参数无效。
b. 错误处理
程序每次报错都有一个,Uncaught xxxError,该错误产生处后面的代码都无法继续执行,这是由于错误未被捕获。
- 捕获错误:try ... catch:
try {
let obj;
console.log(obj.xxx);
} catch (error) {
console.log(error.stack); //TypeError: Cannot read property 'xxx' of undefined at test.html:11
console.log(error.message); //Cannot read property 'xxx' of undefined
}
- 抛出错误:throw error:
function a () {
if (Date.now() % 2 === 1) {
console.log('normal');
} else {
throw new Error('abnormally'); //里面要放message
a();
} catch (error) {
console.log(error)
}
}
}
try {
a();
} catch (error) {
console.log(error)
}
c. 错误对象
- message属性:错误相关信息;
- stack属性:函数调用栈记录信息(里面可定位到出错处)。
二、Promise的理解和使用
1)Promise是什么
- 抽象理解:Promise是js中进行异步编程的新的解决方案,之前是纯回调函数;
- 具体理解:
- 语法上,Promise是一个构造函数;
- 功能上,Promise对象用来封装一个异步操作并可以获取其结果。
2)Promise的状态改变
-
Promise有三个状态:
- pending,待定 初始状态;
- fulfilled,实现,操作成功;
- rejected,被否决,操作失败。
-
状态改变只有两种,这两种情况只要发生,状态就凝固了,不会再变了。
- pending -> fulfilled,得到的结果数据一般称为value;
- pending -> rejected,得到的结果数据一般称为reason。
3)基本使用
const p = new Promise((resolve, reject) => {
const time = Date.now();
setTimeout(() => {
if (time % 2 === 1) {
resolve('成功的调用,time = ' + time);
} else {
reject('失败的调用,time = ' + time);
}
}, 500);
}).then(
value => {
console.log('Successfully! ',value);
//就是上面resolve传过来的参数
},
reason => {
console.log('Failed! ',reason);
//就是上面reject传过来的参数
}
)
三、为什么要使用Promise
1)指定回调函数的方式更加灵活
- 在之前的纯回调函数处理异步事件时,只能在启动异步事件前指定好回调函数,若在执行完毕之后再指定,则接收不到返回的数据;
- 在Promise处理异步事件时:
- 启动异步任务 => 返回Promise对象 => 给Promise对象绑定回调函数(甚至可以在异步事件完成之后)
2)支持链式调用,可以解决回调地狱问题
回调地狱是什么?
- 回调函数嵌套调用,外部回调函数异步执行的结果是嵌套的回调函数执行的条件;
- 缺点:不便于阅读,不便于异常处理;
doFirstThing(function(firstReturn) {
doSecondThing(firstReturn, function(secondReturn) {
doThirdThing(secondReturn, function(thirdReturn) {
console.log('finalReturn: ', thirdReturn);
}, failureCallback)
}, failureCallback)
}, failureCallback)
- 解决方案:Promise链式调用;
- 终极解决方案:async/await;
链式调用
doFirstThing()
.then(function(firstResult) {
doSecondThing(firstResult);
})
.then(function(secondResult) {
doThirdThing(secondResult);
})
.then(function(thirdResult) {
console.log('finalResult: ', thirdResult);
})
.catch(failureCallback);
async await
- await后面接一个会return new promise的函数并执行它;
- await只能放在async函数里;
async function request() {
try {
const firstResult = await doFirstThing()
const secondResult = await doSecondThing(firstResult);
const thirdResult = await doThirdThing(secondResult);
console.log('finalResult: ', thirdResult);
} catch (error) {
failureCallback(error);
}
}
四、Promise的API
1)Promise构造函数:Promise(excutor) {}
- excutor函数:同步执行
(resolve,reject) => {}
- resolve函数:内部定义成功时调用的函数
value => {}
- reject函数:内部定义失败时调用的函数
reason => {}
- 说明:excutor会在Promise内部立即同步回调,异步操作在执行器中执行;
2)Promise.prototype.then方法:(onResolved, onRejected) => {}
- onResolved函数:成功的回调函数
(value) => {}
- onRejected函数:失败的回调函数
(reason) => {}
- 说明:指定用于得到成功value的成功回调和用于得到失败reason的失败回调,返回一个新的Promise对象;
3)Promise.prototype.catch方法:(onRejected) => {}
- onRejected函数:失败的回调函数
(reason) => {}
- 说明:then()的语法糖,相当于
then(undefied, onRejected)
4)Promise.resolve方法:(value) => {}
- value:成功的数据或Promise对象
- 说明:返回一个成功/失败的Promise对象
5)Promise.reject方法:(reason) => {}
- reason:失败的原因
- 说明:返回一个失败的Promise对象
6)Promise.all方法:(promises) => {}
- promises:包含n个promise的数组
- 说明:返回一个新的promise,只有所有的promise都成功才成功,只要有一个失败了就直接失败,所有都成功的话返回一个数组,失败直接返回失败的reason
7)Promise.race方法:(promises) => {}
- promises:包含n个promise的数组
- 说明:返回一个新的promise,第一个完成的promise的结果状态就是最终的结果状态
运用
new Promise((resolve, reject) => {
setTimeout(() => {
// resolve('Suceeded!');
reject('failed!');
}, 1000)
})
.then(value => {console.log(value)})
.catch(reason => {console.log(reason)})
const p1 = new Promise ((resolve, reject) => {
resolve(1);
})
const p2 = Promise.resolve(2);
const p3 = Promise.reject(3);
p1.then(value => {console.log(value)})
p2.then(value => {console.log(value)})
p3.catch(reason => {console.log(reason)})
// const pAll = Promise.all([p1, p2, p3]);
const pAll = Promise.all([p1, p2]);
pAll.then(
values => {
console.log('all onResolved() ', values);
},
reason => {
console.log('all onRejected() ', reason);
}
)
/*
1
2
3
all onResolved() (2) [1, 2]
failed!
*/
五、Promise的几个关键问题
1)如何改变Promise的状态
- 之前也说过,Promise的状态改变只可能有两种过程,pending -> fulfilled和pending -> rejected;
- resolve(value):若当前是pending则变为fulfilled;
- rejected(reason):若当前是pending则变为rejected;
- 抛出异常(throw xxx):若当前是pending则变为rejected,xxx一般是error
2)一个Promise对象指定多个成功/失败的回调函数,都会调用吗?
- 当Promise改变为对应状态时,每个回调函数都会被调用;
- 注意这里指定多个回调函数,不能直接在后面连续.then(),要分开写两次,不然会形成链式指定回调函数,并不是指定多个回调函数;
p.then(
value => {console.log('onResolved()1 ', value)},
reason => {console.log('onRejected()1 ', reason)}
)
p.then(
value => {console.log('onResolved()2 ', value)},
reason => {console.log('onRejected()2 ', reason)}
)
3)Promise状态改变和执行回调函数的先后顺序
-
谁先谁后都有可能,正常情况下是先指定回调再改变状态,反之也没问题;
-
如何先改变状态再执行回调?
- 在执行器中直接调用resolve(), reject(),不使用异步操作;
- 延迟执行调用then(),如用setTimeOut;
-
上面时候才能得到数据?
- 若先指定回调,则当状态发送改变时回调函数就会被调用,得到数据;
- 若先改变状态,则当指定回调函数时函数立即被调用,得到数据;
4)Promise的回调函数是同步还是异步执行
- 当执行器函数中无异步操作,回调函数也已经指定完毕时,回调函数会立马同步被调用吗?还是要放到事件循环队列中等待被调用?
- 答案是,无论如何Promise的回调函数都是异步调用的,要等同步语句都执行完毕才会去队列中调用。
- 验证方法:
new Promise ((resolve, reject) => {
resolve(1);
}).then(
value => {console.log('fulfilled ', value)},
reason => {console.log('rejected ', reason)}
)
console.log('after Promise')
//after Promise先输出
5)Promise.then()返回的新的Promise的结果状态由什么决定?(也就是链式指定回调函数每次的状态由什么决定)
-
由then()指定的回调函数执行的结果决定:
-
抛出异常:新Promise变为rejected,reason为抛出的异常;
-
返回非Promise的任意值:新Promise变为fulfilled,value为返回的值;
-
返回另一个新Promise:Promise的结果变为此Promise的结果(成功或失败由此Promise决定)
-
注意:若只是调用了函数没有返回值,则为undefined。
-
new Promise((resolve, reject) => {
resolve(1)
}).then(
value => {console.log('onResolved()1 ', value); throw 3},
reason => {console.log('onRejected()1 ', reason)}
).then(
value => {console.log('onResolved()2 ', value)},
reason => {console.log('onRejected()2 ', reason)}
)
/*
onResolved()1 1
onRejected()2 3
*/
new Promise((resolve, reject) => {
reject(2)
}).then(
value => {console.log('onResolved()1 ', value)},
reason => {console.log('onRejected()1 ', reason); return 3;}
).then(
value => {console.log('onResolved()2 ', value)},
reason => {console.log('onRejected()2 ', reason)}
)
/*
onRejected()1 2
onResolved()2 3
*/
new Promise((resolve, reject) => {
reject(2)
}).then(
value => {console.log('onResolved()1 ', value)},
reason => {
console.log('onRejected()1 ', reason);
return new Promise((resolve, reject) => resolve(3));
}
).then(
value => {console.log('onResolved()2 ', value)},
reason => {console.log('onRejected()2 ', reason)}
)
/*
onRejected()1 2
onResolved()2 3
*/
new Promise((resolve, reject) => {
resolve(1)
}).then(
value => {console.log('onResolved()1 ', value)},
reason => {console.log('onRejected()1 ', reason)}
).then(
value => {console.log('onResolved()2 ', value)},
reason => {console.log('onRejected()2 ', reason)}
)
/*
onResolved()1 1
onResolved()2 undefined
*/
6)链式调用中,若回调函数为异步操作,则要将异步部分封装进新的Promise对象并作为返回值
- 之前也说过,当.then()函数返回的是另一个新Promise对象时,这个新Promise对象的状态就是下个.then中Promise的状态;
- 所以若要进行异步操作,则需要封装进一个新的Promise对象,同时也有它自己新的resolve reject函数,而不是一直调用一开始的;
new Promise ((resolve, reject) => {
setTimeout(() => {
console.log('执行任务1(异步)');
resolve(1);
}, 1000)
}).then(
value => {
console.log('任务1结果 ', value);
console.log('执行任务2(同步)');
return 2;
}
).then(
value => {
console.log('任务2结果 ', value);
return new Promise ((resolve, reject) => {
setTimeout(() => {
console.log('执行任务3(异步)');
resolve(3);
}, 1000)
})
}).then(
value => {
console.log('任务3结果 ', value);
})
/*
执行任务1(异步)
任务1结果 1
执行任务2(同步)
任务2结果 2
执行任务3(异步)
任务3结果 3
*/
- 从输出结构可以发现,虽然.then中有异步操作,但整个过程依然是按照严格的串联式执行的,有绝对的顺序;
7)错误穿透
- 刚刚处理的都是一路成功的情况,那如果在一开始或某一环节失败了会怎么样?
new Promise ((resolve, reject) => {
reject(1);
}).then(
value => {
console.log('onResolved()1 ', value);
return 2;
}
).then(
value => {
console.log('onResolved()2 ', value);
}
).then(
value => {
console.log('onResolved()3 ', value);
}
).catch(
reason => {console.log('onRejected()1 ', reason)}
)
//最后输出 onRejected()1 1
- 这里的catch并不是第一个Promise的回调函数,由于第一个Promise就失败了,而中间所有的回调函数都只指定了成功的,所以错误会一直向下穿透,直到找到一个失败回调函数;
- 没有定义失败回调函数的.then()实际上内部相当于写上了
reason => {throw reason}
或者reason => Promise.reject(reason)
,将这个错误传下去;
8)中断Promise链
- 当使用Promise的then链式调用时,想在中间中断,不再调用后面的回调函数;
- 办法:在回调函数中返回一个pending状态的Promise对象,
return new Promise (() => {})
; - 返回的Promise对象的状态决定了下次回调函数是执行成功还是失败的,现在返回一个pending状态 也就是还没有结构的状态,自然下一个Promise也没有结果,就这样中断了。
六、async和await
1)async function name([param[, param[, ... param]]])
- 函数的返回值为Promise对象
- Promise对象的结果由async函数执行的返回值决定
- 即若,函数return的不是一个Promise对象,前面由async的话也会自动会封装成一个Promise对象然后返回;
const func1 = async function () {
return Promise.reject(3) //输出 onRejected() 3
return 2 //输出 onResolved() 2
throw 3 //输出 onRejected() 4
}
const result = func1()
result.then(
value => {
console.log('onResolved() ', value)
},
reason => {
console.log('onRejected() ', reason)
}
)
2)[return_value] = await expression;
-
await只能放到async函数里
-
expression的类型
- await 表达式会暂停当前 async function的执行,等待 Promise 处理完成。若 Promise 正常处理(fulfilled),其回调的resolve函数参数作为 await 表达式的值,继续执行 async function。
- 若 Promise 处理异常(rejected),await 表达式会把 Promise 的异常原因抛出。
- 另外,如果 await 操作符后的表达式的值不是一个 Promise,则返回该值本身。
function func2() {
return new Promise((resolve, reject) => {
setTimeout(() => {
resolve(3)
}, 1000)
})
}
async function func1 () {
try {
const value = await func2()
console.log('onResolved() ',value)
} catch (error) {
console.log('onRejected() ', error)
}
}
func1()
//1s后输出 onRejected() 3
- 实际上 async 函数的返回值都是Promise对象,但用await函数接收到的值,是Promise成功之后的value值(失败则进入异常处理)
七、宏任务与微任务
-
JS 中用来存储待执行回调函数的队列包含2 个不同特定的列队
-
宏列队: 用来保存待执行的宏任务(回调), 比如: 定时器回调 / DOM 事件回调 / ajax 回调
-
微列队: 用来保存待执行的微任务( 回调), 比如: promise 的回调 / MutationObserver 的回调
-
JS 执行时会区别这2 个队列
- JS 引擎首先必须先执行所有的初始化同步任务代码
- 每次准备取出第一个宏任务执行前, 都要将所有的微任务一个一个取出来执行,即在执行宏任务过程中,只要微队列不为空,立马跑去微队列中执行
setTimeout(() => {
console.log('timeout callback1()')
Promise.resolve(3).then(
value => {
console.log('Promise onResolved3()', value);
}
)
}, 0)
setTimeout(() => {
console.log('timeout callback2()')
}, 0)
Promise.resolve(1).then(
value => {
console.log('Promise onResolved1()', value);
}
)
Promise.resolve(2).then(
value => {
console.log('Promise onResolved2()', value);
}
)
/*
Promise onResolved1() 1
Promise onResolved2() 2
timeout callback1()
Promise onResolved3() 3
timeout callback2()
*/
八、面试题
- 一般方法都是:
- 找出同步语句,按照出现顺序输出
- 每输出一个同步语句,查看是否更新宏队列与微队列中的项
- 依次输出微队列中的项,查看是否更新宏队列与微队列中的项
- 若微队列为空,再依次查看宏队列中的项,查看是否更新宏队列与微队列中的项
- 要注意的点是:
- new Promise中的语句是同步语句,包括.then语句
- 但.then中的回调函数是异步执行的,要放入微队列
1)
const first = () => (new Promise((resolve, reject) => {
console.log(3)
let p = new Promise((resolve, reject) => {
console.log(7)
setTimeout(() => {
console.log(5)
resolve(6)
}, 0)
resolve(1)
})
resolve(2)
p.then((arg) => {
console.log(arg)
})
}))
first().then((arg) => {
console.log(arg)
})
console.log(4)
2)
setTimeout(() => {
console.log('0')
}, 0)
new Promise((resolve, reject) => {
console.log('1')
resolve()
}).then(() => {
console.log('2')
new Promise((resolve, reject) => {
console.log('3')
resolve()
}).then(() => {
console.log('4')
}).then(() => {
console.log('5')
}).then(() => {
console.log('9')
}).then(() => {
console.log('10')
})
}).then(
() => {console.log('6')},
() => {console.log('error')}
).then(() => {console.log(11)})
new Promise((resolve, reject) => {
console.log('7')
resolve()
}).then(() => {
console.log('8')
})
答案
1)3 7 4 1 2
2)1 7 2 3 8 4 6 5 11 9 10 0
- 其中第二题很可能容易错,一开始我也想错了,主要是.then的执行时间问题
- .then执行是同步的,但.then内部的回调函数的执行要等到上面都有结果了 才能决定执行onResolved还是onRejected
- 这里第一个Promise的第二个.then的执行时间,是第二个Promise里的resolve执行完毕的时候,因为这时第一个Promise的excutor已执行完,Promise实际上已经变为fulfilled的状态了,自然 6 也可以放入微队列了
九、回调函数转Promise(微信小程序)
起因
- 微信小程序作为新兴技术之一,对于es6的支持还没有想象中的好,因此异步的过程都还是采用回调函数的方式,并没有使用Promise,回调地狱问题就粗来了...
- 比如说,我们要获取token,首先调用 wx.login() ,并在传入success回调函数进行后续获取token的操作。获取token还是要调用 wx.request() 函数来发送http请求,又要在这里里面传入success回调函数处理获取的token... 就如下面这样的框架所示:
wx.login({
success: res => {
wx.request({
url: 'url',
success: res => {
// 这里写对获取的token要进行的操作
}
})
}
})
- 以上还只是两次回调的过程,多次回调将难以想象,所以就自己来改成Promise吧!
改变
这里看到,由于本身微信小程序拿到请求数据的过程就是在回调函数里,所以我们就算改 第一次拿到数据还是要在success回调函数里调用resolve拿到数据,重点是后面怎么做;
- 请求函数里返回 new Promise() 之后的数据,把要发送的请求放到这个Promise的excutor里;
- 在success里调用promise对象的resolve函数将返回数据传递出去;
- 由于该函数返回的是一个Promise对象,因此在后续调用的时候,可以直接 .then() 里面获取到刚刚传递出去的数据;
- 若这个后面还有继续调用,就在第三步的函数里直接返回那个 .then() 就行,要传递下去的数据在 相应的函数(resolve reject)里return,后面都能拿到,没必要在每次都包一层new Promise,因为 .then返回的就是promise;
- 框架大概如下:
// 对微信小程序原始请求的封装
function wxPromise1 () {
return new Promise((resolve, reject) => {
wx.request({
url: 'url',
success: res => {
const code = res.code.toString()
// 这里是因为 只要完成了请求 不管是成功还是失败 都会调用success回调函数
// 所以要根据返回的code码来判断请求是否成功
if (code.startsWith('2')) {
resolve(res.data)
} else if (code.startsWith('4')) {
reject()
}
}
})
})
}
// 调用上面的函数
// 后续不用用 new Promise 包装了
function wxPromise2 () {
return wxPromise1().then(res => {
// ...要进行的操作...
return res
})
}
function wxPromise3 () {
return wxPromise2().then(res => {
// ...要进行的操作...
// 这里拿到的res就是上面wxPromise2在resolve里返回的res
return res
})
}