# Promise.all
Promise.all
使用、原理实现及错误处理
# Promise
概念
Promise
是JS
异步编程中的重要概念,异步抽象处理对象,是目前比较流行Javascript
异步编程解决方案之一。Promise.all()
接受一个由promise
任务组成的数组,可以同时处理多个promise
任务,当所有的任务都执行完成时,Promise.all()
返回resolve
,但当有一个失败reject
,则返回失败的信息,即使其他promise
执行成功,也会返回失败。和后台的事务类似。和rxjs
中的forkJoin方
法类似,合并多个 Observable
对象 ,等到所有的 Observable
都完成后,才一次性返回值。
# Promise.all
如何使用
对于 Promise.all(arr)
来说,在参数数组中所有元素都变为决定态后,然后才返回新的 promise
。
// 以下 demo,请求两个 url,当两个异步请求返还结果后,再请求第三个 url
const p1 = request(`http://some.url.1`)
const p2 = request(`http://some.url.2`)
Promise.all([p1, p2])
.then((datas) => { // 此处 datas 为调用 p1, p2 后的结果的数组
return request(`http://some.url.3?a=${datas[0]}&b=${datas[1]}`)
})
.then((data) => {
console.log(msg)
})
1
2
3
4
5
6
7
8
9
10
2
3
4
5
6
7
8
9
10
# Promise.all
原理实现
function promiseAll(promises){
return new Promise(function(resolve,reject){
if(!Array.isArray(promises)){
return reject(new TypeError("argument must be anarray"))
}
var countNum=0;
var promiseNum=promises.length;
var resolvedvalue=new Array(promiseNum);
for(var i=0;i<promiseNum;i++){
(function(i){
Promise.resolve(promises[i]).then(function(value){
countNum++;
resolvedvalue[i]=value;
if(countNum===promiseNum){
return resolve(resolvedvalue)
}
}, function(reason){
return reject(reason)
})
})(i)
}
})
}
var p1=Promise.resolve(1),
p2=Promise.resolve(2),
p3=Promise.resolve(3);
promiseAll([p1,p2,p3]).then(function(value){
console.log(value)
})
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
# Promise.all
错误处理
有时候我们使用Promise.all()
执行很多个网络请求,可能有一个请求出错,但我们并不希望其他的网络请求也返回reject
,要错都错,这样显然是不合理的。如何做才能做到promise.all
中即使一个promise
程序reject
,promise.all
依然能把其他数据正确返回呢?
- 全部改为串行调用(失去了
node
并发优势) - 当
promise
捕获到error
的时候,代码吃掉这个异常,返回resolve
,约定特殊格式表示这个调用成功了
var p1 =new Promise(function(resolve,reject){
setTimeout(function(){
resolve(1);
},0)
});
var p2 = new Promise(function(resolve,reject){
setTimeout(function(){
resolve(2);
},200)
});
var p3 = new Promise(function(resolve,reject){
setTimeout(function(){
try{
console.log(XX.BBB);
}
catch(exp){
resolve("error");
}
},100)
});
Promise.all([p1, p2, p3]).then(function (results) {
console.log("success")
console.log(results);
}).catch(function(r){
console.log("err");
console.log(r);
});
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27