本帖最后由 empyrealtear 于 2025-1-12 21:57 编辑
简易异步线程池
1. 参考文献
2. 修改后代码
const asyncPool = async (arr, process = (item, arr) => item, begin = (v) => v, end = (v) => v,
poolLimit = 10, percentComplete = (pct) => pct) => {
let ret = []
let executing = new Set()
let arr_res = new Array(arr.length)
let completeCount = 0
// begin
percentComplete(completeCount / arr.length * 100)
arr = begin(arr)
// process
for (let [index, item] of arr.entries()) {
var p = Promise.resolve().then(async () => {
try {
let res = await process(item, arr)
arr_res[index] = res
} catch (err) {
console.warn(err)
arr_res[index] = err
}
return
}).finally(() => {
percentComplete((++completeCount) / arr.length * 100)
})
ret.push(p)
executing.add(p)
let clean = () => executing.delete(p)
p.then(clean).catch(clean)
if (executing.size >= poolLimit) {
await Promise.race(executing)
}
}
// end
return Promise.all(ret).then(() => {
percentComplete(completeCount / arr.length * 100)
return end(arr_res)
})
}
3. 使用样例
fetch(url).then(async (xhr) => {
let data = xhr.response?.data
if (data) {
return await asyncPool(data, async (v, arr) => {
let row = {}
await fetch(v.url).then((xhr2) => {
let data2 = xhr2.response?.data
row = { value: data2 }
})
return row
})
}
})