`fs.appendFile()` 的原子性保证是什么?

2024-01-09

的文档fs.appendFile()关于原子性保证含糊不清。

我特别想知道如果在同一个文件上对数据进行两次或多次调用,而不等待其间的回调,是否可以交错或删除数据(例如,由于写入重叠区域)。例如,采用以下代码:

fs.writeFileSync('blort.bin', '');
for (let i = 0; i < 10; i++) {
  const data = Buffer.alloc(65536, i);
  fs.appendFile('blort.bin', data, () => {});
}

假设没有文件系统错误并且没有其他进程写入相关文件,那么在完成所有追加操作后,Node 会做出以下保证吗?:

  • 该文件的大小正好是 640k。
  • 该文件由 10 个 64k 数据块组成,每个数据块统一包含一个字节值。

我也很好奇是否有任何序列化保证(尽管我认为没有)。也就是说,追加是否保证按顺序执行?

更新:添加了说明,表明这是作用于该文件的唯一代码。


据我所知,您的追加将按顺序进行,除非另一个进程正在写入目标文件,否则不会覆盖现有数据。

节点是单线程的(从正在运行的应用程序的角度来看)并在内部对文件写入进行排队,因此如果您编写1 2 and 3到一个文件,该文件将包含123.

Node 确实会启动新线程来处理 IO,但这是运行时的一个功能,对开发人员是隐藏的。

UPDATE

在第一次阅读时,我错过了您的要求,即您的附加发生在回调为空白或不起作用的多个调用中。

这极大地改变了事情。

如果您使用异步调用,例如fs.appendFile(),您必须提供一个必须检查错误的回调。如果您堆叠异步调用而不等待它们的回调结束,无法保证您的写入会按顺序发生或不会覆盖以前的数据。

底线:如果没有始终检查错误的适当回调,请勿使用异步函数。

否则就会破坏 Node 事件驱动模型的整个理念。

我通常解释这一点的方式是考虑到,在异步调用中,你无法预测回调何时被触发由于许多因素,包括 cpu 和 IO 负载。

如果您无法知道何时触发回调,则无法预测多个调用将按顺序运行。

UPDATE 2

需要明确的是,您无法在不提供回调的情况下调用异步函数并保证any预期的行为。

从事件的角度来思考:

fs.appendFile()将事件处理程序添加到 IO 线程,并将包含其参数的事件发送到 IO 线程。

操作完成后,将触发此事件处理程序,这要么是您的回调,要么将触发您的回调。

如果不等待操作完成,您就无法保证操作已成功或已完成您所期望的操作。

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

`fs.appendFile()` 的原子性保证是什么? 的相关文章

随机推荐