javascript - 为什么 Promise-Limit 在 Nodejs 中不起作用?

标签 javascript node.js concurrency promise limit

我有大量图像想要下载。我正在使用 request-promise 包从其 URL 下载图像。由于存在大量图像,服务器过载,下载会挂起或损坏,因此我决定使用 promise-limit库来设置并发限制。这是我的代码:


const fs = require('fs');
const request = require('request-promise');
const promiseLimit = require('promise-limit');

var limit = promiseLimit(30);

async function download(data) {
    console.log(data);

    return Promise.all(data.map( (obj) => {
        return limit(() => downloadRequest(obj))
    })).then(() => {
        console.log("All images downloaded.")
    })

function downloadRequest(obj) {
    var img = obj.dest;
    var url = obj.url;
    var req = request(url);
    req.pipe(fs.createWriteStream(img));
}

我完全按照 github 页面中给出的方式复制了示例,但该方法在没有满足 Promise.all() 的情况下返回。我不明白我做错了什么,我认为 Promise.all() 肯定会等到它解决所有 promise 。

在后端,此调用用作

...
.then((data) => {
        return downloader.download(data);
    })
    .then(() => {
        var filename = "Sample.pdf";
// Pages.json is written early in the chain, contains URL and corresponding image paths
        return generator.generate('./Pages.json', filename); 
    });

这是否意味着 NodeJS 已经在尝试从 Pages.json 生成文件?如何让这部分代码同步下载?

最佳答案

您的函数downloadRequest()不会返回 promise 。它必须返回一个与 contains 中的异步操作相关联的 Promise,以便在异步操作完成时解决 Promise,或者在异步操作出现错误时拒绝 Promise。只有当它这样做时,limit() 包才能正确完成其工作。

由于您使用流并将其传输到 downloadRequest() 中,因此您必须手动构造一个 Promise,然后监视流中的各种事件以了解它何时完成或有一个错误,以便您可以解决或拒绝该 promise 。

这里有一个如何让 downloadRequest() 正确返回 promise 的想法:

function downloadRequest(obj) {
    return new Promise((resolve, reject) => {
        const img = obj.dest;
        const url = obj.url;
        const req = request(url);

        req.on('error', reject);

        const ws = fs.createWriteStream(img);
        ws.on('error', reject);
        req.pipe(ws).on('finish', resolve);
    });
}

并且,现在建议使用 pipeline() 函数而不是 .pipe(),因为它可以在错误情况下进行更完整的清理,并且还有一个该内置的 promise 版本:

const { pipeline } = require('stream/promises');

function downloadRequest(obj) {
    return pipeline(request(obj.url), fs.createWriteStream(obj.dest));
}

附注如果您不知道,request() 库已被弃用,建议您不要再在新项目中使用它。有一个可供选择的替代库列表 here ,所有这些也都有内置的 Promise 支持。我查看了各种选择,尝试了几个并决定使用 got()在我的工作中。

关于javascript - 为什么 Promise-Limit 在 Nodejs 中不起作用?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/65747873/

相关文章:

swift - 从后台操作在 NSView 中绘制无效

javascript - 带有 if 语句的each() 不起作用

javascript - 填充 vuetify 从 json 数组中选择

node.js - 使用 NPM 安装包时,你能告诉它使用其依赖项之一的不同版本吗?

android - 未处理的 promise 拒绝警告 : Unhandled promise rejection (rejection id: 1): Error: spawn EACCES Ionic 3

multithreading - 为什么我可以将 "move"静态 &str 放入 Rust 的多个线程中?

javascript - 当鼠标悬停在容器的 1/3 宽度上时使内容可见,而不添加子容器

javascript 未插入数组

node.js - Discord.JS 可以添加角色但无法删除角色?

Java同步-不正确的发布