gpt4 book ai didi

javascript - 如何在处理输出时暂停和取消暂停 Node 对象流

转载 作者:搜寻专家 更新时间:2023-10-31 23:17:34 24 4
gpt4 key购买 nike

我目前正在通过发出 'line' 事件的转换流来逐行处理文件流。我希望能够在发现当前行符合某些条件后暂停输入文件流,开始处理新流,并在完成后逐行恢复处理原始流。我已将其浓缩为以下最小示例:

测试.咖啡:

fs = require 'fs'    
TestTransform = require './test-transform'
inStream = new TestTransform
fs.createReadStream("./test.coffee").pipe(inStream)
inStream.on 'line', (line) ->
process.stdout.write "-->"
if line.match /line\.match/g
process.stdout.write line
console.error "PAUSE"
inStream.pause()
fs.createReadStream("./test.coffee").pipe(process.stdout).on 'end', ->
console.error "UNPAUSE"
inStream.resume()
else
process.stdout.write line

test-transform.coffee:

Transform = require('stream').Transform

module.exports =
class TestTransform extends Transform
constructor: ->
Transform.call @, readableObjectMode: true
@buffer = ""

pushLines: ->
newlineIndex = @buffer.indexOf "\n"
while newlineIndex isnt -1
@push @buffer.substr(0, newlineIndex + 1)
@emit 'line', @buffer.substr(0, newlineIndex + 1)
@buffer = @buffer.substr(newlineIndex + 1)
newlineIndex = @buffer.indexOf "\n"

_transform: (chunk, enc, cb) ->
@buffer = @buffer + chunk.toString()
@pushLines()
cb?()

_flush: (cb) ->
@pushLines()
@buffer += "\n" # ending newline
@push @buffer
@emit 'line', @buffer # push last line
@buffer = ""
cb?()

(不要太担心Transform流,它只是一个例子。)无论如何,coffee test.coffee的输出看起来像:

-->fs = require 'fs'
-->
-->TestTransform = require './test-transform'
-->
-->inStream = new TestTransform
-->
-->fs.createReadStream("./test.coffee").pipe(inStream)
-->
-->inStream.on 'line', (line) ->
--> process.stdout.write "-->"
--> if line.match /line\.match/g
PAUSE
--> process.stdout.write line
--> console.error "PAUSE"
--> inStream.pause()
--> fs.createReadStream("./test.coffee").pipe(process.stdout).on 'end', ->
--> console.error "UNPAUSE"
--> inStream.unpause()
--> else
--> process.stdout.write line
-->
fs = require 'fs'

TestTransform = require './test-transform'

inStream = new TestTransform

fs.createReadStream("./test.coffee").pipe(inStream)

inStream.on 'line', (line) ->
process.stdout.write "-->"
if line.match /line\.match/g
process.stdout.write line
console.error "PAUSE"
inStream.pause()
fs.createReadStream("./test.coffee").pipe(process.stdout).on 'end', ->
console.error "UNPAUSE"
inStream.unpause()
else
process.stdout.write line

很明显,管道没有暂停,它只是继续直到完成(即使 PAUSE 正在按预期运行),并且因为 "UNPAUSE" 也永远不会被写出,'end' 回调永远不会触发。将流切换为从转换流暂停/取消暂停到 readStream 似乎也不起作用。我从这种行为中假设 Node 流以某种方式不遵守事件回调中的暂停/取消暂停。

可能还有另一种方法可以在不调用暂停/取消暂停的情况下完成此操作;如果有某种方式喜欢等待流结束并暂停当前执行线程,那将有效地完成我正在尝试做的事情。

最佳答案

如果我没有正确理解问题,这里有一个使用 Dust.js 的简单 Node 应用程序解决了这个问题。

Dust 是一个模板引擎,但它最好的特性之一是它对 Node Streams 的原生理解。此示例使用 Dust 2.7.0。

我正在使用 node-byline作为 Transform 流的替代品,但它做同样的事情——逐行读取流。

var fs = require('fs'),
byline = require('byline'),
dust = require('dustjs-linkedin');

var stream = byline(fs.createReadStream('./test.txt', { encoding: 'utf8' }));

var template = dust.loadSource(dust.compile('{#byline}--> {.|s}{~n}{match}{/byline}'));

dust.stream(template, {
byline: stream,
match: function(chunk, context) {
var currentLine = context.current();

if(currentLine.match(/line\.match/g)) {
return fs.createReadStream('./test.txt', 'utf8');
}
return chunk;
}
}).pipe(process.stdout);

这是我程序的输出:

$ node index.js
--> fs = require 'fs'
--> TestTransform = require './test-transform'
--> inStream = new TestTransform
--> fs.createReadStream("./test.coffee").pipe(inStream)
--> inStream.on 'line', (line) ->
--> process.stdout.write "-->"
--> if line.match /line\.match/g
fs = require 'fs'
TestTransform = require './test-transform'
inStream = new TestTransform
fs.createReadStream("./test.coffee").pipe(inStream)
inStream.on 'line', (line) ->
process.stdout.write "-->"
if line.match /line\.match/g
process.stdout.write line
console.error "PAUSE"
inStream.pause()
fs.createReadStream("./test.coffee").pipe(process.stdout).on 'end', ->
console.error "UNPAUSE"
inStream.resume()
else
process.stdout.write line

--> process.stdout.write line
--> console.error "PAUSE"
--> inStream.pause()
--> fs.createReadStream("./test.coffee").pipe(process.stdout).on 'end', ->
--> console.error "UNPAUSE"
--> inStream.resume()
--> else
--> process.stdout.write line

如您所见,它正确地交错了输出。如果我可以进一步详细说明 Dust 部分的工作原理,请告诉我。

编辑:这是对 Dust 模板的具体解释。

{#byline} {! look for the context variable named `byline` !}
{! okay, it's a stream. For each `data` event, output this stuff once !}
-->
{.|s} {! output the current `data`. Use |s to turn off HTML escaping !}
{~n} {! a newline !}
{match} {! look up the variable called `match` !}
{! okay, it's a function. Run it and insert the result !}
{! if the result is a stream, stream it in. !}
{/byline} {! done looping !}

关于javascript - 如何在处理输出时暂停和取消暂停 Node 对象流,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/29721851/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com