我编写了一个命令行实用程序,该实用程序可以生成大量数据,然后将生成的数据转储到一个大文件中。
我注意到一些竞争条件问题,其中进程读取文件,并同时将另一个文件添加到该文件中以读取/写入相同内容。这会导致某些数据从最终文件中删除/覆盖,因为当一个进程中添加的内容合并生成的内容时,该数据不会读入另一个进程中。
我目前正在研究使用NodeJs互斥库来尝试向进程添加一些读/写锁。我还可以尝试将子进程重构为单进程异步任务,但这会大大降低操作速度,并且需要大量的额外工作。
我研究了async-mutex
和lockfile
来完成此任务。
但是,它们似乎仅在一个进程中起作用,并且该锁未在所有子进程之间共享(这很有意义),从而导致每个子进程都获得自己的锁。
我正在尝试对async-mutex
const definition = await generateData();
await mutex.runExclusive(() => {
console.log('LOCK ACQUIRED FOR', uuid);
if (fs.existsSync(baseFile)) {
baseDefinition = yaml.safeLoad(fs.readFileSync(baseFile, 'utf8'));
}
baseDefinition = _.merge(baseDefinition, definition);
fs.writeFileSync(baseFile, yaml.safeDump(baseDefinition));
console.log('RELEASING LOCK FOR', uuid);
});
const moreData = await generateMoreData();
await mutex.runExclusive(async () => {
console.log('SECOND LOCK ACQUIRED FOR', uuid);
await moveData(moreData, baseFile);
console.log('SECOND LOCK RELEASED FOR', uuid);
});
generateData
和generateMoreData
方法会花费一些时间,所以我不想在生成过程中保持锁定(这就是为什么单线程异步/等待方法会花费更多时间的原因。时间来完成,以及为什么要生成多个子进程)。 moveData
方法对baseFile
进行了一些其他读/写操作,因此我只是在尝试运行该方法之前尝试获取该锁。
运行此命令时,我得到一些类似这样的输出
LOCK ACQUIRED FOR 526ac716-b135-4a2d-986d-75bee856411b
LOCK ACQUIRED FOR 104073b7-6d7f-4ca1-8513-eec103877898
RELEASING LOCK FOR 526ac716-b135-4a2d-986d-75bee856411b
SECOND LOCK ACQUIRED FOR 526ac716-b135-4a2d-986d-75bee856411b
RELEASING LOCK FOR 104073b7-6d7f-4ca1-8513-eec103877898
SECOND LOCK ACQUIRED FOR 104073b7-6d7f-4ca1-8513-eec103877898
SECOND LOCK RELEASED FOR 526ac716-b135-4a2d-986d-75bee856411b
LOCK ACQUIRED FOR 155eb578-bc91-4bee-830f-bab2e1363bb8
SECOND LOCK RELEASED FOR 104073b7-6d7f-4ca1-8513-eec103877898
LOCK ACQUIRED FOR 8e6e08cf-8b3b-4040-af88-86670ad5cbb1
RELEASING LOCK FOR 155eb578-bc91-4bee-830f-bab2e1363bb8
SECOND LOCK ACQUIRED FOR 155eb578-bc91-4bee-830f-bab2e1363bb8
LOCK ACQUIRED FOR cb79f948-e4a4-4f75-ba31-fa2b743ce886
RELEASING LOCK FOR 8e6e08cf-8b3b-4040-af88-86670ad5cbb1
SECOND LOCK ACQUIRED FOR 8e6e08cf-8b3b-4040-af88-86670ad5cbb1
SECOND LOCK RELEASED FOR 155eb578-bc91-4bee-830f-bab2e1363bb8
RELEASING LOCK FOR cb79f948-e4a4-4f75-ba31-fa2b743ce886
SECOND LOCK ACQUIRED FOR cb79f948-e4a4-4f75-ba31-fa2b743ce886
SECOND LOCK RELEASED FOR 8e6e08cf-8b3b-4040-af88-86670ad5cbb1
SECOND LOCK RELEASED FOR cb79f948-e4a4-4f75-ba31-fa2b743ce886
如您所见,由于锁的获取和释放是彼此独立的,因此显然锁不会在进程之间共享。
NodeJ中是否有一种方法可以独立于文件的运行进程来锁定文件?