有人可以解释以下行为吗?这肯定是由于异步I / O,但下面的代码基于许多简单的例子,其中一些来自SO,显然没有讨论这里的问题,其中流没有按预期读取。
解决方案是什么?我试图从第一原理中理解潜在的问题,所以请不要建议我使用已发布的npm stream->字符串包。感谢。
给定文件n.js
'use strict';
const streamToString = (s, cb) => {
const chunks = []
s.on('readable', () => {
console.log('data')
let chunk
while( null !== (chunk = s.read())) {
chunks.push(chunk)
}
})
s.on('end', () => {
console.log('end')
cb(Buffer.concat(chunks).toString())
})
}
let o
const f = (str) => {
o = JSON.parse(str)
}
const fs = require('fs')
const rs = fs.createReadStream('t.json')
streamToString(rs, f)
console.log('o = ' + o)
在终端
$ uname -a
Linux bsh 4.10.0-40-generic #44~16.04.1-Ubuntu SMP Thu Nov 9 15:37:44 UTC 2017 x86_64 x86_64 x86_64 GNU/Linux
$ node --version
v6.12.0
$ node n.js
o = undefined
data
data
end
输出适用于任何非空输入文件,包含简单验证的JSON。
我也尝试过'读'事件,即
s.on('read', (chunk) => {
console.log('data')
chunks.push(chunk)
})
,输出
o = undefined
data
end
答案 0 :(得分:0)
1:console.log('o = ' + o)
是同步代码,因此它会在您传递给执行异步的f
的回调函数streamToString
之前运行。因此,当时间console.log('o = ' + o)
完成时,f
函数尚未执行,这就是o
未定义的原因。只需移动回调函数console.log
中的f
即可获得所需内容。
const f = (str) => {
o = JSON.parse(str);
console.log('o = ' + o);
}
2:readable
在整个过程中发出两次,read
只发出一次,详见here