JavaScript基本原理(1):程序异步-基于Promise/async的实现
Promise
Promise.resolve(x) 可以看作是 new Promise(resolve => resolve(x)) 的简写,可以用于快速封装字面量对象或其他对象,将其封装成 Promise 实例
- 返回一个Promise对象,使用该对象来注册处理结果和错误的回调
- 同时可以串联.then 和 .catch
本质
- 最终要被交付的结果的容器
- 可以注册监听器的对象(指运行resolve之后能够按需执行下来的代码)
例子
function addAsync(x, y) {
return new Promise(
(resolve, reject) => { // (A)
if (x === undefined || y === undefined) {
reject(new Error('Must provide two parameters'));
//将这个error对象传递给.catch
} else {
resolve(x + y);
//将值传递给then
}
});
}
addAsync(3, 4)
.then(result => { // success
assert.equal(result, 7);
})
.catch(error => { // failure
assert.fail(error);
});
- 一个 Promise 中包含几个状态
- Pending
- fulfilled
- rejected
- 而resolve和reject的执行则会导致这些状态的变化
- 具体可以参考手写实现Promise
- 个人觉得是写的最好的手写实现代码和相关解释了
.then
- 返回一个新的Promise
回调返回一个非Promise值
- 直接作为值传递给下一个.then()
Promise.resolve('abc')
.then(str => {
return str + str; // (A)
})
.then(str2 => {
assert.equal(str2, 'abcabc'); // (B)
});
回调返回一个Promise
Promise.resolve('abc')
.then(str => {
return Promise.resolve(123); // (A)
})
.then(num => {
assert.equal(num, 123);
});
- 个人理解:替换原本执行 .then()创建的Promise而使用当前返回的这个值作为返回的Promise
- 非Promise返回值时,原本的.then()会对返回结果进行封装后返回
抛出异常
.catch
- 和then的区别仅有,.catch由reject触发,将其回调的操作转换为Promise
Promise.resolve()
- 创建一个被给定值履行的Promise
- 即一般Promise创建后状态默认为 pending
- 而以这样方式创建的promise状态为fulfill
Promise.resolve(123)
.then(x => {
//此处传入的x就是123,因为运行来resolve所以直接执行then
});
Promise.reject()
- 大概同上
Promise的优势
- 回调上更加清晰
- 链式调用更加清晰
举例
- 对于需要获得返回文本的网络请求
传统的回调函数
import * as fs from 'fs';
fs.readFile('person.json',
(error, text) => {
if (error) { // (A)
// Failure
assert.fail(error);
} else {
// Success
try { // (B)
const obj = JSON.parse(text); // (C)
assert.deepEqual(obj, {
first: 'Jane',
last: 'Doe',
});
} catch (e) {
// Invalid JSON
assert.fail(e);
}
}
});
使用Promise优化
readFileAsync('person.json')
.then(text => { // (A)
// Success
const obj = JSON.parse(text);
assert.deepEqual(obj, {
first: 'Jane',
last: 'Doe',
});
})
.catch(err => { // (B)
// Failure: file I/O error or JSON syntax error
assert.fail(err);
});
//此处可以考虑一下如何实现 readFileAsync()
- http
async/await
async : 异步的缩写
await : async wait 的缩写
await只能出现在async中
async返回值是一个包含返回结果的promise对象
获取返回值的方法
-
使用then链来处理对象
-
使用await提取
-
await : 等待的内容,等待async完成的返回值,或者等待一个表达式的计算结果
- 若返回值是一个Promise对象,阻塞后面的代码,等待promise对象resolve,然后将resolve的值作为await表达式的运算结果返回
从上面的描述中,我们不难看出
await 实际上就是.then 的写法
而 async 则就是声明一个promise对象
方法
Promise.all()
Promise.all可以将多个Promise实例包装成一个新的Promise实例,同时执行多个异步任务
同时,成功和失败的返回值是不同的,成功的时候返回的是一个结果数组,而失败的时候则返回最先被reject失败状态的值。
let p1 = new Promise((resolve, reject) => {
resolve('成功了')
})
let p2 = new Promise((resolve, reject) => {
resolve('success')
})
let p3 = Promise.reject('失败')
let p4 = Promise.reject('fail')
Promise.all([p1, p2]).then((result) => {
console.log(result) //['成功了', 'success']
}).catch((error) => {
console.log(error)
})
Promise.all([p1,p4,p3,p2]).then((result) => {
console.log(result)
}).catch((error) => {
console.log(error) // 失败了,打出 '失败'
})
返回的数组顺序和Promise接收到的数据顺序一致,即使p2比p1先获得结果,在数组中,p1还是在p2的前面
Promise.race()
和all()类似,接收Promise集合作为但是race中成功和失败状态都是返回最先获得的那个
async
题目描述:
代码执行时,立即输出 0,之后每隔 1 秒依次输出 1,2,3,4,循环结束后在大概第 5 秒的时候输出 5(这里使用大概,是为了避免钻牛角尖的同学陷进去,因为 JS 中的定时器触发时机有可能是不确定的
// 模拟其他语言中的 sleep,实际上可以是任何异步操作
const sleep = (timeountMS) => new Promise((resolve) => {
setTimeout(resolve, timeountMS);
});
(async () => { // 声明即执行的 async 函数表达式
for (var i = 0; i < 5; i++) {
await sleep(1000);
console.log(new Date, i);
}
await sleep(1000);
console.log(new Date, i);
})();