9

下面的代码创建一个文本文件,然后读取它,覆盖它,然后再次读取它。除了创建文件之外,三个 I/O 操作都是使用 Node.js 异步 readFile 和 writeFile 执行的。

我不明白为什么第一次读取没有返回错误也没有数据。这段代码的输出是:

  • 开始...
  • 完毕。
  • 首次读取返回 EMPTY 数据!
  • 写完OK
  • 第二次读取返回的数据:更新的文本

即使操作以任意顺序发生(由于它们的异步性质),我也不希望得到一个“空数据”对象。

任何想法为什么我在读取文件时得到一个空数据(并且没有错误)?

我可以做些什么来确保读取文件内容吗?

var fs = require('fs');
var fileName = __dirname + '/test.txt';

// Create the test file (this is sync on purpose)
fs.writeFileSync(fileName, 'initial test text', 'utf8');


console.log("Starting...");

// Read async
fs.readFile(fileName, 'utf8', function(err, data) {
    var msg = "";
    if(err)
        console.log("first read returned error: ", err);
    else {
        if (data === null) 
            console.log("first read returned NULL data!");
        else if (data === "") 
            console.log("first read returned EMPTY data!");
        else
            console.log("first read returned data: ", data);
    }
});


// Write async
fs.writeFile(fileName, 'updated text', 'utf8', function(err) {
    var msg = "";
    if(err)
        console.log("write finished with error: ", err);
    else
        console.log("write finished OK");
});


// Read async
fs.readFile(fileName, 'utf8', function(err, data) {
    var msg = "";
    if(err)
        console.log("second read returned error: ", err);
    else
        if (data === null) 
            console.log("second read returned NULL data!");
        else if (data === "") 
            console.log("second read returned EMPTY data!");
        else
            console.log("second read returned data: ", data);
});


console.log("Done.");
4

2 回答 2

18

您的代码要求竞争条件。您的第一次同步写入可能正在写入文件,但随后您的第一次读取、第二次写入和第二次读取同时放入事件循环中。

这里可能发生了什么?第一次读取从文件系统获得读取权限,第二次写入从文件系统获得写入权限并立即将文件归零以供将来更新,然后第一次读取读取现在为空的文件。然后第二次写入开始写入数据,第二次读取在完成之前不会获得读取权限。

如果你想避免这种情况,你需要使用流程:

fs.writeFileSync(filename, 'initial', 'utf8');
fs.readFile(filename, 'utf8', function(err, data) {
    console.log(data);
    fs.writeFile(filename, 'text', 'utf8', function(err) {
        fs.readFile(filename, 'utf8', function(err, data) {
            console.log(data);
        });
    });
});

如果那个“金字塔”侮辱了你的编程敏感性(为什么不呢?)使用async libraryseries功能:

fs.writeFileSync(filename, 'initial', 'utf8');
async.series([
    function(callback) {
        fs.readFile(filename, 'utf8', callback);
    },
    function(callback) {
        fs.writeFile(filename, 'text', 'utf8', callback);
    },
    function(callback) {
        fs.readFile(filename, 'utf8', callback);
    }
], function(err, results) {
    if(err) console.log(err);
    console.log(results); // Should be: ['initial', null, 'text']
});

编辑:对于不熟悉async库和现代 Javascript 功能的人来说更紧凑,但也更“神奇”:

fs.writeFileSync(filename, 'initial', 'utf8');
async.series([
    fs.readFile.bind(this, filename, 'utf8'),
    fs.writeFile.bind(this, filename, 'text', 'utf8'),
    fs.readFile.bind(this, filename, 'utf8'),
], function(err, results) {
    if(err) console.log(err);
    console.log(results); // Should be: ['initial', null, 'text']
});

EDIT2:无需查找bind. 第一个参数必须是this对象(或任何您想用作的对象this)。

于 2012-04-29T01:05:38.253 回答
1

我有一个类似的问题。我正在将文本写入文件,并且有一个更改处理程序告诉我文件何时更改,此时我尝试异步读取它以进一步处理文件的新内容。

大多数时候都有效,但在某些情况下,异步读取的回调返回一个空字符串。因此,也许更改事件发生在文件完全写入之前,所以当我尝试读取它时,我得到了空字符串。现在人们可能希望 ASYNC 读取能够识别出文件正在写入过程中,因此应该等到写入操作完成。似乎在 Node.js 中写入不会锁定文件不被读取,因此如果您在写入过程中尝试读取,则会得到意想不到的结果。

我能够通过检测 ASYNC 读取的结果是否为空字符串来解决此问题,如果是,则对同一文件进行额外的 SYNC 读取。这似乎产生了正确的内容。是的 SYNC 读取速度较慢,但​​我只在 ASYNC 读取似乎未能产生预期内容时才这样做。

于 2020-04-17T19:53:23.880 回答