实现一个promise.all()?

思路

  • 接收一个 Promise 实例的数组或具有 Iterator 接口的对象作为参数

  • 这个方法返回一个新的 promise 对象,

  • 遍历传入的参数,用Promise.resolve()将参数"包一层",使其变成一个promise对象

  • 参数所有回调成功才是成功,返回值数组与参数顺序一致

  • 参数数组其中一个失败,则触发失败状态,第一个触发失败的 Promise 错误信息作为 Promise.all 的错误信息。

实现

实现一

function promiseAll(promiseArray) {
    return new Promise(function (resolve, reject) {
        if (!Array.isArray(promiseArray)) {
            throw new TypeError(`argument must be a array`)
        }

        let resolvedCounter = 0;
        let promiseNum = promiseArray.length;
        let resolvedResult = [];

        for (let i = 0; i < promiseNum; i++) {
            Promise.resolve(promiseArray[i]).then(
                value => {
                    resolvedCounter++;
                    resolvedResult[i] = value;
                    if (resolvedCounter == promiseNum) {
                        return resolve(resolvedResult)
                    }
                },
                reason => {
                    return reject(error)
                }
            )
        }
    })
}

//------------------------- test---------------------------
let p1 = new Promise(function (resolve, reject) {
    setTimeout(function () {
        resolve(1)
    }, 1000)
})
let p2 = new Promise(function (resolve, reject) {
    setTimeout(function () {
        resolve(2)
    }, 2000)
})
let p3 = new Promise(function (resolve, reject) {
    setTimeout(function () {
        resolve(3)
    }, 3000)
})
promiseAll([p3, p1, p2]).then(res => {
    console.log(res) // [3, 1, 2]
})

实现二

/**
 * Promise.all
 * @description 当这个数组里的所有promise对象全部变为resolve状态的时候,才会resolve, 当有一个promise对象变为reject状态时,就不再执行直接 reject
 * @param {*} values promise对象组成的数组作为参数
 */

Promise.prototype.myAll = (values) => {
    return new Promise(
        (resolve, reject) => {
            let resultArr = []
            let count = 0
            let resultByKey = (value, index) => {
                resultArr[index] = value
                if (++count === values.length) {
                    resolve(resultArr)
                }
            }
            values.forEach(
                (promise, index) => {
                    promise.then(
                        (value) => {
                            resultByKey(value, index)
                        },
                        reject
                    )
                }
            )
        }
    )

}



all的应用场景

一般来说,Promise.all 用来处理多个并发请求,将一个页面所用到的在不同接口的数据一起请求过来。直到完成所有的promise,才执行all的then方法。

拓展:实现promise队列

Promise.all().then()的用法,all()表示所有的操作均执行完成,再执行then()中的方法。而all()中的函数,不是顺序执行,而是同时执行

如果希望then()之前的函数都是顺序执行的,可以考虑通过reduce函数来实现。

这就实现了一个promise队列

console.log("----------------------------------------")
const arr09 = [
    () => new Promise(
        (resolve) => {
            setTimeout(() => {
                console.log(10);
                resolve(10);
            }, 1000)
        }
    ).then(res => { console.log(res) }),
    () => new Promise(
        (resolve) => {
            setTimeout(() => {
                console.log(20);
                resolve(20);
            }, 1000)
        }
    ).then(res => { console.log(res) }),

    () => new Promise(
        (resolve) => {
            setTimeout(() => {
                console.log(30);
                resolve(30);
            }, 1000)
        }
    ).then(res => { console.log(res) }),

    () => (40)
]

const result09 = arr09.reduce(
    (prev, next) => prev.then(next),
    
    // 这里是为了new一个promise形成链式调用
    Promise.resolve()
)
console.log(result09)


const createPromise = (id) => () => 
	new Promise(resolve=>
		setTimeout(()=>{
			console.log("promise->"+id+":"+new Date());
			resolve();
		},1000)
	)
var tasks = [createPromise(1),createPromise(2),createPromise(3)];
console.log(tasks);
var doTask = tasks.reduce((prev,next)=>prev.then(()=>next()),Promise.resolve());
doTask.then(()=>console.log("all done."));

其它写法

[1, 2, 3, 4, 5].reduce((pre, cur) => {
    return pre.then(() => {
        return new Promise((resolve, reject) => {
            setTimeout(() => {
                console.log(cur)
                resolve()
            }, 1000)
        })
    })
}, Promise.resolve())

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: Promise.all是JavaScript中的一个静态方法,它接受一个Promise对象的数组作为参数,并返回一个新的Promise对象。这个新的Promise对象在所有传入的Promise对象都完成(无论成功还是失败)之后才会完成。如果传入的所有Promise对象都成功完成,则新的Promise对象也会成功完成,否则它会失败。 例如,假设我们有两个异步操作:一个是从服务器获取用户信息,另一个是从服务器获取用户的订单信息。我们可以使用Promise.all来等待这两个操作都完成,然后执行一些操作: ``` Promise.all([getUserInfo(), getOrderInfo()]).then(function(results) { // results is an array with the results of the two promises // do something with the results }); ``` 如果你想自己实现Promise.all,你可以这样做: ``` function myPromiseAll(promises) { return new Promise(function(resolve, reject) { // track the number of promises that have resolved var resolvedCount = 0; // array to store the results of the promises var results = []; // iterate through the promises promises.forEach(function(promise, index) { // resolve the promise promise.then(function(result) { // store the result results[index] = result; // increment the count resolvedCount++; // if all promises have resolved, resolve the main promise if (resolvedCount === promises.length) { resolve(results); } }).catch(function(error) { // if any of the promises fail, reject the main promise reject(error); }); }); }); } ``` 这个函数接受一个Promise对象的数组,并返回一个新的Promise对象。它会依次解决所有 ### 回答2: Promise.all是一个用于将多个Promise对象包装成一个新的Promise对象的方法。当所有的Promise都成功完成时,新的Promise对象才会成功,如果有任何一个Promise失败,则新的Promise对象将失败。 下面是一个实现Promise.all的简单例子: ```javascript function promiseAll(promises) { return new Promise((resolve, reject) => { let results = []; let count = 0; promises.forEach((promise, index) => { Promise.resolve(promise).then((result) => { results[index] = result; count++; if(count === promises.length) { resolve(results); } }).catch((error) => { reject(error); }); }); }); } ``` 我们首先创建一个新的Promise对象,并且定义一个空数组`results`用于保存每个Promise的结果,以及一个计数器`count`用于记录成功完成的Promise数量。 然后遍历传入的Promise数组,将每个Promise都转换为resolved状态的Promise对象,然后使用`then`方法获取其结果,将结果保存在`results`数组中,并将`count`增加1。 当所有的Promise都成功完成时,也就是`count`等于传入的Promise数量时,我们调用`resolve`方法并传入`results`数组,使新的Promise对象进入resolved状态。 如果有任何一个Promise失败,我们则直接调用`reject`方法将新的Promise对象置为rejected状态。 综上所述,上面的函数就是一个简单的实现Promise.all的例子。 ### 回答3: 实现一个`Promise.all`是指在一个时间点同时对多个promise进行并行处理,并且等待所有promise都完成后才返回结果。我们可以使用原生JavaScript实现这个功能。 首先,我们需要创建一个包装函数,接收一个promise数组作为参数,返回一个新的promise。然后,我们使用`Promise.all`的形式迭代每个传入的promise,并使用`Promise.resolve`确保每个值都是promise对象。 在迭代过程中,我们可以使用`Promise.all`返回的新promise的`then`方法来等待所有的promise都成功后再执行回调函数。在这个回调函数中,我们可以将所有promise的结果以数组的形式进行返回。 如果其中一个promise被拒绝(rejected),我们可以使用catch方法来捕获拒绝的promise并将其返回。这样就可以确保即使有一个promise被拒绝,整个`Promise.all`函数仍然可以正常运行。 下面是一个示例的实现代码: ```javascript function promiseAll(promises) { return new Promise((resolve, reject) => { let results = []; let completedPromises = 0; const totalPromises = promises.length; if (totalPromises === 0) { resolve(results); } for (let i = 0; i < totalPromises; i++) { Promise.resolve(promises[i]).then((result) => { results[i] = result; completedPromises++; // 所有的promise都已完成,返回结果数组 if (completedPromises === totalPromises) { resolve(results); } }).catch((error) => { reject(error); }); } }); } ``` 使用上述函数,我们可以通过传入一个promise数组来处理多个异步操作,并在所有promise都成功完成后获取结果。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值