在等待Promise.all()之后,我很难理解为什么我还有悬而未决的承诺。
在下面的示例中,我通过使用.map()
对数组的每个元素调用异步函数来创建一个承诺数组。
现在,为什么承诺仍然是悬而未决的呢?我现在的理解是:
then()
的承诺得到解决,storeData()
就会触发storeData()
,newDataArray
就会解析newDataArray
中的所有承诺,或者一旦第一个承诺被拒绝,就会返回promisesArray
。storeData(OldDataArray).then(values => console.log(values))
// console shows:
// { id: 1, data: Promise { <pending> } },
// { id: 2, data: Promise { <pending> } }
const storeData = async (OldDataArray) => {
try {
const promisesArray = OldDataArray.map((item) => {
let newData = downloadMoreDetails(item.id, item.group); //async function, see below
return {
id: item.id,
data: newData,
};
});
const newDataArray = await Promise.all(promisesArray); // <-- I'm awaiting all promises to complete before assigning to newDataArray
return newDataArray;
} catch (error) {
console.log(error)
}
};
const downloadMoreDetails = async (id, group) => {
const response = await fetch(
`example.com/id/group.xml`
);
if (!response.ok) {
throw new Error(`HTTP error ${response.status}`);
}
const str = await response.text();
const json = convert.xml2json(str, {
compact: true,
spaces: 2,
});
return json;
};
发布于 2021-10-07 16:28:27
newData
是一个承诺,但你没有等待承诺。相反,您正在等待一个对象数组( {id: item.id, data: newData }
),该数组中包含了它的承诺。Promise.all()
不会在这些对象内部寻找承诺,然后等待承诺。它只看到一个简单的对象数组,这意味着它没有什么可做的。您可以通过这样做来解决这个问题:
const storeData = async (OldDataArray) => {
try {
const promisesArray = OldDataArray.map(async (item) => {
let newData = await downloadMoreDetails(item.id, item.group); //async function, see below
return {
id: item.id,
data: newData,
};
});
return Promise.all(promisesArray);
} catch (error) {
// log and rethrow error so the caller gets the rejection
console.log(error);
throw error;
}
};
这将.map()
回调更改为async
。做两件有益的事。首先,这意味着来自.map()
的结果数组将是一个承诺数组,因为async
回调总是返回一个承诺。其次,它允许在回调中使用await
,这样就可以用实际的数据填充返回的对象,而不是承诺。
然后,来自return
回调内部的async
将导致该值成为async
函数返回的承诺的解析值。
注意,您也可以在不添加像这样的async/await
的情况下完成它:
const storeData = (OldDataArray) => {
const promisesArray = OldDataArray.map((item) => {
return downloadMoreDetails(item.id, item.group).then(newData => {
return {
id: item.id,
data: newData,
};
});
});
return Promise.all(promisesArray).catch(error => {
// log and rethrow error so the caller gets the rejection
console.log(error);
throw error;
});
};
在这个版本中,您直接从.map()
回调返回一个承诺,并确保该承诺解析为您的数据对象。
https://stackoverflow.com/questions/69489072
复制相似问题