Node.js流可以作为协程进行制作吗?

Muq*_*ith 2 generator coroutine node.js

有没有办法使Node.js流作为协程.

斐波那契数字流的示例.

fibonacci.on('data', cb);
//The callback (cb) is like
function cb(data)
{
    //something done with data here ...
}
Run Code Online (Sandbox Code Playgroud)

期望

function* fibonacciGenerator()
{
    fibonacci.on('data', cb);
    //Don't know what has to be done further... 
};

var fibGen = fibonacciGenerator();
fibGen.next().value(cb);
fibGen.next().value(cb);
fibGen.next().value(cb);
.
.
.
Run Code Online (Sandbox Code Playgroud)

从发电机中取出所需的数字.这里Fibonacci数字系列只是一个例子,实际上流可以是任何文件,mongodb查询结果等.

也许这样的事情

  1. 将'stream.on'函数设为生成器.
  2. 将yield放在回调函数中.
  3. 获取生成器对象.
  4. 调用next并在流中获取下一个值.

是否至少可能,如果是,如果不是为什么?也许一个愚蠢的问题:)

noi*_*ixy 6

如果您不想使用转换器(例如Babel)或等到async/ await将其转到Node.js,您可以使用生成器和承诺自己实现它.

缺点是你的代码必须存在于生成器中.



首先,您可以创建一个接收流的帮助程序并返回一个函数,该函数在被调用时返回流的下一个"事件"的承诺(例如data).

function streamToPromises(stream) {
  return function() {
    if (stream.isPaused()) {
      stream.resume();
    }

    return new Promise(function(resolve) {
      stream.once('data', function() {
        resolve.apply(stream, arguments);
        stream.pause();
      });
    });
  }
}
Run Code Online (Sandbox Code Playgroud)

当你不使用它时它会暂停流,并在你问它下一个值时恢复它.


接下来,你有一个帮助器接收一个生成器作为参数,每次它产生一个promise,它解析它并将其结果传递回生成器.

function run(fn) {
  var gen = fn();
  var promise = gen.next().value;

  var tick = function() {
    promise.then(function() {
      promise = gen.next.apply(gen, arguments).value;
    }).catch(function(err) {
      // TODO: Handle error.
    }).then(function() {
      tick();
    });
  }

  tick();
}
Run Code Online (Sandbox Code Playgroud)

最后,您将在生成器中执行自己的逻辑,并使用run帮助程序运行它,如下所示:

run(function*() {
  var nextFib = streamToPromises(fibonacci);

  var n;

  n = yield nextFib();
  console.log(n);

  n = yield nextFib();
  console.log(n);
});
Run Code Online (Sandbox Code Playgroud)
  • 您自己的生成器将产生promise,暂停其执行并将控件传递给run函数.
  • run函数将解析promise并将其值传递回您自己的生成器.

这是它的要点.您还需要修改streamToPromises以检查其他事件(例如enderror).