繁体   English   中英

Node.js 中的异步函数

[英]Async Function in Node.js

我的 Node.js API 中有一个端点,它返回由 Google-Search- scraper库提供的结果的 JSON 数组。

app.get('/google_image_search', (req, res) => {
    var options = {
        query: 'grenouille',
        age: 'y', // last 24 hours ([hdwmy]\d? as in google URL)
        limit: 10,
        params: {} // params will be copied as-is in the search URL query string
    };

    var results = [];
    scraper.search(options, function(err, url, meta) {
        sem.take(function() { 
            if(err) throw err;

            var result = {
                title: meta.title,
                meta: meta.meta,
                description: meta.desc
            }
            results.push(result);
            sem.leave();
        });
    })

    console.log(results);

    res.json({
        results
    });
})

我需要 console.log(results) 和 res.json({ results }) 在 scraper.search 功能完成后发生。 它目前总是返回一个空数组。

为每个结果调用传递给 scraper.search() 函数的函数。 因此,如果有 10 个结果该函数运行 10 次,这就是为什么我要等到数组已满才能发送响应。

我曾尝试在不同的地方使用信号量和互斥锁,但没有成功。 任何建议表示赞赏。


这是使用 LIMIT 变量检查我的结果数组来解决的。 在下面标记为正确的答案中概述。

感谢大家的投入。


我需要 console.log(results) 和 res.json({ results }) 在 scraper.search 功能完成后发生。

把它放在scraper.search()的最内层回调中。

scraper.search(options, function(err, url, meta) {
        if(err) throw err;

        var result = {
            title: meta.title,
            meta: meta.meta,
            description: meta.desc
        };
        results.push(result);
        console.log(result);
        res.json({results});
});

每次运行回调时都会调用console.log()res.json() 如果您只想在 10 个结果或其他结果后执行此操作,请添加代码以检查条件并仅在正确的时间运行console.log()和/或res.json()

您还可以查看诸如async / await ,但鉴于您发布的代码,以上可能是最增量的解决方案。

您现在拥有console.log()res.json()在于,它将异步回调使用函数视为同步函数。

Trott 的答案在正确的轨道上,但是如何拥有一个每次递增的变量,然后当它等于 10(或 9,取决于您的实现方式)时,运行您的完成代码。 您也可以只计算数组中的元素。

app.get('/google_image_search', (req, res) => {
    var options = {
        query: 'grenouille',
        age: 'y', // last 24 hours ([hdwmy]\d? as in google URL)
        limit: 10,
        params: {} // params will be copied as-is in the search URL query string
    };

    var results = [];
    scraper.search(options, function(err, url, meta) {
        sem.take(function() { 
            if(err) throw err;

            var result = {
                title: meta.title,
                meta: meta.meta,
                description: meta.desc
            }
            results.push(result);
            sem.leave();
        });
        if(results.length==10) {
            console.log(results);

            res.json({
                results
            });
        }
    })
})

res.send放在回调之外将导致与此问题类似的竞争条件。 google-search-scraper库的一个缺点是它不是为了收集结果而设计的。

这应该是固定的:

var LIMIT = 10;
var options = { limit: LIMIT, ... };

var results = [];
var errs = [];
var resultsCount = 0;

function resultsHandler() {
    if (errs.length) {
       // handle error
    } else
       res.json({ results });
}

scraper.search(options, function resultHandler(err, url, meta) {
    if (err)
        errs.push(err);
    else {
        var result = {
            title: meta.title,
            meta: meta.meta,
            description: meta.desc
        };

        results.push(result);
    });

    resultsCount++;

    if (resultsCount === LIMIT)
       resultsHandler();
});

如果search可能在某些条件下不调用回调,这将不起作用。

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM