2

我希望我问的问题是正确的,但本质上,NodeJS中有没有办法让CommonJS模块在(长)计算后返回数据流而不是最终的数据块?假设有不止一种方法,那么这些技术是什么?

例如,假设我有一个findPrimes函数,写成CommonJS模块:

查找素数.js

/**
 * @module  findPrimes
 * @param {int} n - Find all primes less than this number
 * @return {array}
 */
module.exports = function(n) {
    if (n < 2) {
        return [];
    } else if (n === 2) {
        return [2];
    }

    var primes = [2];

    for (let i = 3; i < n; i += 2) {
        let is_prime = true;
        let sq = Math.ceil(Math.sqrt(i));

        for (let t = 2; t <= sq; t++) {
            if (i % t === 0) {
                is_prime = false;
                break;
            }
        }

        if (is_prime) {
            primes.push(i);
        }
    }

    return primes;
};

如您所见,此函数返回一个包含所有小于其输入的素数的数组。它在计算完所有这些数字返回数组。

所以,假设我在节点脚本中使用这个模块

index.js

const primes = require('./find-primes.js');

// Usage: `node index.js <num>`
let primes_less_than = process.argv[2];

console.log(primes(primes_less_than));

当我使用 arg 运行上述脚本时25,我得到以下(预期的)输出:

$ node index.js 25
[ 2, 3, 5, 7, 11, 13, 17, 19, 23 ]

但是,假设我通过了一个更大的数字,比如 10,000,000

$ node index.js 10000000
# Takes a while to run before outputting the numbers...

虽然这可行,但理想情况下,我希望程序在完成之前开始写出它计算的数字。

所以我的程序仍然需要一段时间才能运行,但它开始向屏幕输出信息的速度比“先计算一切,然后输出所有结果”要快得多。

达到这种效果的最佳方法是什么?承诺

我愿意接受任何和所有的技术,谢谢。

4

2 回答 2

3

您必须使用 nodejs 中的 Stream 类,并且该方法应使用流导出回调,如评论所述:

const stream = require('stream');

function getPrimesStream(n, cb) {
    const primesStream = new stream.Stream();
    cb(null, primesStream);
    if (n >= 2) {
        primesStream.emit('data', 2);
    }

    const primes = [2];

    for (let i = 3; i < n; i += 2) {
        let isPrime = true;
        const sq = Math.ceil(Math.sqrt(i));

        for (let t = 2; t <= sq; t += 1) {
            if (i % t === 0) {
                isPrime = false;
                break;
            }
        }

        if (isPrime) {
            primesStream.emit('data', i);
            primes.push(i);
        }
    }

    return primes;
}

getPrimesStream(1000, function (err, stream) {
    stream.on('data', function (data) {
        console.log(data);
    });
    stream.on('end', function() {
        console.log('finished');
    });
});
于 2017-06-08T19:46:42.980 回答
1

您可以为此使用 Node.js 的stream模块。几行代码很难回答您的问题,但如果您真的对流的工作原理感兴趣,请观看此视频(这是我在慕尼黑的 Node.js 聚会上谈论 Node.js 流, 德国)。

或者,您可以使用带有 的生成器函数yield,但这也很难用几行从头开始解释。

无论如何,生成器函数是您应该寻找的术语。

于 2017-06-08T19:37:03.277 回答