【问题标题】:Read and Write files from 'fs' not working asynchronously从“fs”读取和写入文件不能异步工作
【发布时间】:2022-01-19 19:48:01
【问题描述】:

我正在尝试构建一个读取和写入一些文件的脚本。这些功能工作正常,但我无法异步读取/写入功能。我遇到的主要问题是我正在尝试编写一个文件,然后我正在尝试导入它。 例如:给定这两个函数,第一个函数写入./myFile.js,第二个函数尝试像这样导入文件:

var fs = require('fs');
const myFile = require('./myFile')

async function writeMyFile() {
    return new Promise((resolve, reject) => {
        fs.promises.readFile('./myFile.js', 'utf8').then((file) => {

        const replacedData = file + 'newLine'

            fs.promises
                .writeFile('./myFile.js', replacedData)
                .then(() => resolve('Writted'))
                .catch(() => reject('Error'));
        });
    });
}

function importMyFile() {
    console.log(myFile) // this is always empty
}

await writeMyFile()
importMyFile()

问题是我可以看到myFile 被正确写入,但我在importMyFile 函数中看不到它。当我分别尝试这些功能时,它可以工作(执行第一个,然后执行第二个)

我尝试了以下方法:

  • 使用fs 库中的promise 函数:fs.promises.readFile()fs.promises.writeFile()(上例)
  • 返回一个在writeMyFile 结束时解析的 Promise。 (上例)
  • 直接在readMyFile函数中导入文件如下:
function importMyFile() {
    const myFile = require('./myFile')
    console.log(myFile) // this is always empty
}
  • 无需像这样返回新的 Promise:
async function writeMyFile() {
    const file = await fs.promises.readFile('./myFile.js', 'utf8')

    const replacedData = file + 'newLine'

    await fs.promises.writeFile('./myFile.js', replacedData)
}

我还看到在控制台中,日志的顺序是这样的:

'Writted'
{} //empty file

请记住,这是我案例的简化版本。

编辑我尝试写入和导入的文件看起来总是这样:

module.exports = []

它被填充了,它看起来像这样:

const myModule = {
    name: 'name',
    surname: 'surname'
}

module.exports = [myModule]

我希望有人知道这里发生了什么。谢谢!

【问题讨论】:

  • 你不需要异步和承诺。 async 为您做出承诺,因此在做出承诺之后等待完成,但在写入完成之前完成
  • 我已经尝试过不返回 Promise -> await fs.promises.readFile(...)await fs.promises.writeFile(..) 但没有任何变化
  • 我也不知道require 会返回任何有用的东西。而且,该文件是否已经存在?为什么要先读再写?如果它不存在,它将不会被写入
  • 文件始终存在。我将澄清我的代码示例。
  • require 返回module.exports。如果您在 myFile.js 中没有任何导出,它将返回一个空对象。

标签: javascript node.js asynchronous async-await fs


【解决方案1】:

使用requirereadFile 不同。使用 require 将 .js 文件作为模块加载,并引入其导出的元素以在您的代码中使用。如果您只是对打印 .js 文件中的文本感兴趣,请使用 fs.readFileSync 执行此操作(如果您使用调用的同步版本,则不需要 promise)。

您的代码会将字符串“newLine”添加到文件末尾,但如果您的意图是添加一个实际的换行符,则应该添加“\n”。

您的代码的更简单版本可能如下所示:

function writeMyFile() {
  const file = fs.readFileSync('./myFile.js', 'utf8');
  const replacedData = file + '\n';
  fs.writeFileSync('./myFile.js', replacedData);
  return 'Written';
}

function importMyFile() {
  // actually read the file instead of using require()
  const file = fs.readFileSync('./myFile.js', 'utf8');
  console.log(file);
}

writeMyFile()
importMyFile()

由于没有进行异步调用,因此不需要承诺或等待。

如果您确实打算使用内部的 .js 代码,并且使用 require 很重要,请注意,一旦您以这种方式将其加载到内存中,磁盘上的文件更改后它不会被刷新。即使您随后致电require,它也会很聪明地说“哦,我已经在内存中拥有那个”并且不会重新加载它。因此,延迟调用require('./myFile'),直到进行任何必要的更改并将require 语句放在importMyFile() 中。

【讨论】:

  • 好了.. 你明白了 :) 我在我的脚本中不止一次地导入了 './myFile'。因此,即使我将它导入到函数中,它也会从内存中加载它。非常感谢:)
【解决方案2】:
await writeMyFile()
importMyFile()

必须在异步函数中。

例如,像这样调用它们,这应该可以工作。


(async ()=>{

await writeMyFile()
importMyFile()

})()

还有输出,{} 表示没有从 myfile.js 导出模块。

编辑: “myFile”变量包含旧数据。因为它是在执行开始时导入的。 importMyFile 应该重新导入 myfile.js。


function importMyFile() {
    myFile = require('./myFile')
    console.log(myFile)  
}

在此之后,writeMyFile 的新更改将显示在控制台中,如果函数更改了 myfile.js 的导出。

【讨论】:

  • 它们已经在异步函数中了。
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 2015-12-14
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多