因為實現(xiàn)不了Promise.all,一場面試涼涼了
作者:前端胖頭魚
來源:SegmentFault 思否社區(qū)
前言
Promise.resolve
簡要回顧
Promise.resolve(value) 方法返回一個以給定值解析后的 Promise 對象。
如果這個值是一個 promise ,那么將返回這個 promise ;
如果這個值是thenable(即帶有 "then" 方法),返回的promise會“跟隨”這個thenable的對象,采用它的最終狀態(tài);否則返回的promise將以此值完成。
Promise.resolve最終結果還是一個Promise,并且與Promise.resolve(該值)傳入的值息息相關
傳入的參數(shù)可以是一個Promise實例,那么該函數(shù)執(zhí)行的結果是直接將實例返回
這里最主要需要理解跟隨,可以理解成Promise最終狀態(tài)就是這個thenable對象輸出的值
// 1. 非Promise對象,非thenable對象
Promise.resolve(1).then(console.log) // 1
// 2. Promise對象成功狀態(tài)
const p2 = new Promise((resolve) => resolve(2))
Promise.resolve(p2).then(console.log) // 2
// 3. Promise對象失敗狀態(tài)
const p3 = new Promise((_, reject) => reject('err3'))
Promise.resolve(p3).catch(console.error) // err3
// 4. thenable對象
const p4 = {
then (resolve) {
setTimeout(() => resolve(4), 1000)
}
}
Promise.resolve(p4).then(console.log) // 4
// 5. 啥都沒傳
Promise.resolve().then(console.log) // undefined
源碼實現(xiàn)
Promise.myResolve = function (value) {
// 是Promise實例,直接返回即可
if (value && typeof value === 'object' && (value instanceof Promise)) {
return value
}
// 否則其他情況一律再通過Promise包裝一下
return new Promise((resolve) => {
resolve(value)
})
}
// 測試一下,還是用剛才的例子
// 1. 非Promise對象,非thenable對象
Promise.myResolve(1).then(console.log) // 1
// 2. Promise對象成功狀態(tài)
const p2 = new Promise((resolve) => resolve(2))
Promise.myResolve(p2).then(console.log) // 2
// 3. Promise對象失敗狀態(tài)
const p3 = new Promise((_, reject) => reject('err3'))
Promise.myResolve(p3).catch(console.error) // err3
// 4. thenable對象
const p4 = {
then (resolve) {
setTimeout(() => resolve(4), 1000)
}
}
Promise.myResolve(p4).then(console.log) // 4
// 5. 啥都沒傳
Promise.myResolve().then(console.log) // undefined
Promise.reject
簡要回顧
Promise.reject(new Error('fail'))
.then(() => console.log('Resolved'),
(err) => console.log('Rejected', err))
// 輸出以下內(nèi)容
// Rejected Error: fail
// at <anonymous>:2:16
源碼實現(xiàn)
Promise.myReject = function (value) {
return new Promise((_, reject) => {
reject(value)
})
}
// 測試一下
Promise.myReject(new Error('fail'))
.then(() => console.log('Resolved'),
(err) => console.log('Rejected', err))
// Rejected Error: fail
// at <anonymous>:9:18
Promise.all
簡要回顧
const p = Promise.all([p1, p2, p3])
const p1 = Promise.resolve(1)
const p2 = new Promise((resolve) => {
setTimeout(() => resolve(2), 1000)
})
const p3 = new Promise((resolve) => {
setTimeout(() => resolve(3), 3000)
})
const p4 = Promise.reject('err4')
const p5 = Promise.reject('err5')
// 1. 所有的Promise都成功了
const p11 = Promise.all([ p1, p2, p3 ])
.then(console.log) // [ 1, 2, 3 ]
.catch(console.log)
// 2. 有一個Promise失敗了
const p12 = Promise.all([ p1, p2, p4 ])
.then(console.log)
.catch(console.log) // err4
// 3. 有兩個Promise失敗了,可以看到最終輸出的是err4,第一個失敗的返回值
const p13 = Promise.all([ p1, p4, p5 ])
.then(console.log)
.catch(console.log) // err4
源碼實現(xiàn)
Promise.myAll = (promises) => {
return new Promise((rs, rj) => {
// 計數(shù)器
let count = 0
// 存放結果
let result = []
const len = promises.length
if (len === 0) {
return rs([])
}
promises.forEach((p, i) => {
// 注意有的數(shù)組項有可能不是Promise,需要手動轉(zhuǎn)化一下
Promise.resolve(p).then((res) => {
count += 1
// 收集每個Promise的返回值
result[ i ] = res
// 當所有的Promise都成功了,那么將返回的Promise結果設置為result
if (count === len) {
rs(result)
}
// 監(jiān)聽數(shù)組項中的Promise catch只要有一個失敗,那么我們自己返回的Promise也會失敗
}).catch(rj)
})
})
}
// 測試一下
const p1 = Promise.resolve(1)
const p2 = new Promise((resolve) => {
setTimeout(() => resolve(2), 1000)
})
const p3 = new Promise((resolve) => {
setTimeout(() => resolve(3), 3000)
})
const p4 = Promise.reject('err4')
const p5 = Promise.reject('err5')
// 1. 所有的Promise都成功了
const p11 = Promise.myAll([ p1, p2, p3 ])
.then(console.log) // [ 1, 2, 3 ]
.catch(console.log)
// 2. 有一個Promise失敗了
const p12 = Promise.myAll([ p1, p2, p4 ])
.then(console.log)
.catch(console.log) // err4
// 3. 有兩個Promise失敗了,可以看到最終輸出的是err4,第一個失敗的返回值
const p13 = Promise.myAll([ p1, p4, p5 ])
.then(console.log)
.catch(console.log) // err4
// 與原生的Promise.all返回是一致的
Promise.allSettled
簡要回顧
const p1 = Promise.resolve(1)
const p2 = new Promise((resolve) => {
setTimeout(() => resolve(2), 1000)
})
const p3 = new Promise((resolve) => {
setTimeout(() => resolve(3), 3000)
})
const p4 = Promise.reject('err4')
const p5 = Promise.reject('err5')
// 1. 所有的Promise都成功了
const p11 = Promise.allSettled([ p1, p2, p3 ])
.then((res) => console.log(JSON.stringify(res, null, 2)))
// 輸出
/*
[
{
"status": "fulfilled",
"value": 1
},
{
"status": "fulfilled",
"value": 2
},
{
"status": "fulfilled",
"value": 3
}
]
*/
// 2. 有一個Promise失敗了
const p12 = Promise.allSettled([ p1, p2, p4 ])
.then((res) => console.log(JSON.stringify(res, null, 2)))
// 輸出
/*
[
{
"status": "fulfilled",
"value": 1
},
{
"status": "fulfilled",
"value": 2
},
{
"status": "rejected",
"reason": "err4"
}
]
*/
// 3. 有兩個Promise失敗了
const p13 = Promise.allSettled([ p1, p4, p5 ])
.then((res) => console.log(JSON.stringify(res, null, 2)))
// 輸出
/*
[
{
"status": "fulfilled",
"value": 1
},
{
"status": "rejected",
"reason": "err4"
},
{
"status": "rejected",
"reason": "err5"
}
]
*/
不管是全部成功還是有部分失敗,最終都會進入Promise.allSettled的.then回調(diào)中
最后的返回值中,成功和失敗的項都有status屬性,成功時值是fulfilled,失敗時是rejected
最后的返回值中,成功含有value屬性,而失敗則是reason屬性
源碼實現(xiàn)
Promise.myAllSettled = (promises) => {
return new Promise((rs, rj) => {
let count = 0
let result = []
const len = promises.length
// 數(shù)組是空的話,直接返回空數(shù)據(jù)
if (len === 0) {
return resolve([])
}
promises.forEach((p, i) => {
Promise.resolve(p).then((res) => {
count += 1
// 成功屬性設置
result[ i ] = {
status: 'fulfilled',
value: res
}
if (count === len) {
rs(result)
}
}).catch((err) => {
count += 1
// 失敗屬性設置
result[i] = {
status: 'rejected',
reason: err
}
if (count === len) {
rs(result)
}
})
})
})
}
// 測試一下
const p1 = Promise.resolve(1)
const p2 = new Promise((resolve) => {
setTimeout(() => resolve(2), 1000)
})
const p3 = new Promise((resolve) => {
setTimeout(() => resolve(3), 3000)
})
const p4 = Promise.reject('err4')
const p5 = Promise.reject('err5')
// 1. 所有的Promise都成功了
const p11 = Promise.myAllSettled([ p1, p2, p3 ])
.then((res) => console.log(JSON.stringify(res, null, 2)))
// 輸出
/*
[
{
"status": "fulfilled",
"value": 1
},
{
"status": "fulfilled",
"value": 2
},
{
"status": "fulfilled",
"value": 3
}
]
*/
// 2. 有一個Promise失敗了
const p12 = Promise.myAllSettled([ p1, p2, p4 ])
.then((res) => console.log(JSON.stringify(res, null, 2)))
// 輸出
/*
[
{
"status": "fulfilled",
"value": 1
},
{
"status": "fulfilled",
"value": 2
},
{
"status": "rejected",
"reason": "err4"
}
]
*/
// 3. 有兩個Promise失敗了
const p13 = Promise.myAllSettled([ p1, p4, p5 ])
.then((res) => console.log(JSON.stringify(res, null, 2)))
// 輸出
/*
[
{
"status": "fulfilled",
"value": 1
},
{
"status": "rejected",
"reason": "err4"
},
{
"status": "rejected",
"reason": "err5"
}
]
*/
Promise.race
簡單回顧
const p = Promise.race([p1, p2, p3])
const p1 = new Promise((resolve, reject) => {
setTimeout(resolve, 500, 1)
})
const p2 = new Promise((resolve, reject) => {
setTimeout(resolve, 100, 2)
})
Promise.race([p1, p2]).then((value) => {
console.log(value) // 2
})
Promise.race([p1, p2, 3]).then((value) => {
console.log(value) // 3
})
源碼實現(xiàn)
Promise.myRace = (promises) => {
return new Promise((rs, rj) => {
promises.forEach((p) => {
// 對p進行一次包裝,防止非Promise對象
// 并且對齊進行監(jiān)聽,將我們自己返回的Promise的resolve,reject傳遞給p,哪個先改變狀態(tài),我們返回的Promise也將會是什么狀態(tài)
Promise.resolve(p).then(rs).catch(rj)
})
})
}
// 測試一下
const p1 = new Promise((resolve, reject) => {
setTimeout(resolve, 500, 1)
})
const p2 = new Promise((resolve, reject) => {
setTimeout(resolve, 100, 2)
})
Promise.myRace([p1, p2]).then((value) => {
console.log(value) // 2
})
Promise.myRace([p1, p2, 3]).then((value) => {
console.log(value) // 3
})
結尾

評論
圖片
表情
