掌握 Promise 的逻辑方法
Promise 是 ES2015 新增的对象
Promise 对象有几个组合方法,可以将多个承诺合并成一个进行处理
分别是 Promise.all, Promise.race, Promise.allSettled, Promise.any
这些方法都可以接收一组承诺,返回一个新的承诺
Promise.all(values)
其中参数 values 是一个可迭代对象,比如数组
在后文中我都使用词语“成功”表示承诺 resolve,“失败”表示承诺 reject
Promise.all
Promise.all 方法返回的承诺会等到参数中所有的承诺都成功之后才会成功,只要其中有一个失败了则返回的承诺也会立即失败,不会等到那些还挂起的承诺有结果
Promise.all 方法可以用来处理那些缺一不可的逻辑
示例:同时发出多个请求都成功后才能进行下一步
const coffee = fetch('/coffee')
const tea = fetch('/tea')
const me = fetch('/me')
try {
// 我全都要
const res = await Promise.all([coffee, tea, me])
// res 是个数组
} catch (err) {
// 至少有一个请求失败了
}
Promise.all 方法 resolve 的结果是一个数组
参数中的承诺对象并不会按顺序解析,但是结果数组中的顺序与传递顺序相同
Promise.race
Promise.race 方法返回参数中最快的那个承诺,如果最快的那个承诺成功则返回的承诺也会成功,如果最快的那个承诺失败则返回的承诺也是失败,那些还没有结果的会被忽略
示例:给一个复杂任务设定一个超时时间
// 设置一个定时器,时间到了就 reject 一个承诺
const timeout = new Promise((resolve, reject) => {
setTimeout(reject, 3000)
})
// 这是一个复杂任务,可能是轮询,也可能是多个异步请求
const missions = new Promise((resolve, reject) => { ... })
try {
const res = await Promise.race([timeout, missions])
// missions 任务在 3 秒内成功了
} catch () {
// 时间超过 3 秒了或者任务失败了
}
Promise.allSettled
Promise.allSettled 方法返回的承诺对象会等到参数中所有的承诺对象都完成后才成功,无论怎样该方法返回的承诺都不会失败
和 Promise.all 方法的区别
Promise.all 方法需要参数中的所有承诺都成功
而 Promise.allSettled 对参数中的承诺是成功还是失败并不关心,只要有结果就行
示例:一次性上传多个文件,其中上传成功和上传失败的互不影响,在一轮上传任务完成之后,可以筛选出那些上传失败的重新上传
// 这是一个上传文件的方法,返回一个承诺
const upload = file => {
const formData = new FormData()
formData.append('file', file)
return fetch('/upload', {
method: 'POST',
body: formData
})
}
// 在文件选择框的 change 事件处理程序中执行上传任务
document.querySelector('input[type="file"]').addEventListener('change', function(e) {
if (!e.target.value) return // 如果啥都没有选
const files = e.target.files
// 遍历所有文件并上传返回承诺数组
const promises = files.map(file => upload(file))
// 数组正好可以传递给 allSettled 方法
const res = await Promise.allSettled(promises)
// 全部上传任务都完成了,挑选出上传失败的重新上传
})
该方法是 ES2020 新添加的方法
Promise.any
Promise.any 方法返回一组承诺中最快成功的那个承诺,如果参数中所有承诺都失败了,那么返回的承诺也失败
和 Promise.race 方法的区别
Promise.race 返回参数中最快的那个承诺,无论它是成功还是失败
而 Promise.any 关注的是参数中最快成功的那个承诺
和 Promise.all 方法的区别
Promise.any 和 Promise.all 是完全相反的
Promise.any 参数中全部承诺都失败了才会失败,Promise.all 参数中全部承诺都成功了才会成功
Promise.any 参数中一旦有一个承诺成功了返回的新承诺就会成功,Promise.all 参数中一旦有一个承诺失败了返回的新承诺就会失败
示例:同时请求几张图像,但是我们只需要用到其中的一张,就可以用 Promise.any 方法挑选出最先加载成功的那张
// 这是一个请求图像的方法,成功则返回图像
const fetchImg = async (url) => {
return fetch(url).then(res => {
if (!res.ok) {
throw new Error('HTTP error!')
} else {
return res.blob()
}
})
}
cosnt img1 = fetchImg('/1.png')
const img2 = fetchImg('/2.png')
try {
const res = await Promise.any([img1, img2])
// 只要有一个成功就可以
const url = URL.createObjectURL(res)
const img = document.createElement('img')
img.src = url
document.body.appendChild(img)
} catch () {
// 一个都没加载成功 QAQ
}
该方法还处于草案中,目前最新的 Chrome, Firefox, Safari 支持
以上所有方法要么环境本身已经实现,要么在 core.js 中有实现,都可以放心食用
本文来自博客园,作者:by.Genesis,转载请注明原文链接:https://www.cnblogs.com/xyzhanjiang/p/14090357.html