首页 > 解决方案 > Node.js 使用 async/await 读取文件

问题描述

我正在尝试使用异步/等待代码来读取文件。这是我的代码:

var fs = require('fs');

function readFile(fileName) {
  return new Promise(resolve => {
    //console.log(test);
    fs.readFile(fileName, 'utf8', function (err, data) {
      if (err) throw err;

      console.log(fileName)
      console.log(data)
    })
    resolve();
  });
}

async function run() {
  await readFile('file1.txt');
  await readFile('file2.txt');
  readFile('file3.txt');
}

run();

但结果仍然是随机的。这意味着 file3 有时会在 file2 之前读取。我在哪里做错了?

标签: node.jsasync-await

解决方案


有很多方法可以实现这一目标。

其中大部分在此链接中进行了解释


我写一个简单的:

1)使用util.promisify将回调方法转换为promise:

const fs = require('fs');
const util = require('util');
const readFile = (fileName) => util.promisify(fs.readFile)(fileName, 'utf8');

(async () => {
  try {
    const files = ['file1.txt', 'file2.txt', 'file3.txt'];
    for (const file of files) {
      console.log(
        await readFile(file)
      );
    }
  }
  catch (error) {
    console.error(error);
  }
})();

2)*Sync方法。由于您的代码不处理并发您可以使用*Sync方法:

const fs = require('fs');

try {
  const files = ['file1.txt', 'file2.txt', 'file3.txt'];
  for (const file of files) {
    console.log(
      fs.readFileSync(file, 'utf8')
    );
  }
}
catch (error) {
  console.error(error);
}


顺便提一句。这是您的固定代码:

var fs = require('fs');

function readFile(fileName) {
  return new Promise((resolve, reject) => {
    fs.readFile(fileName, 'utf8', function (error, data) {
      if (error) return reject(error);

      console.log(fileName)
      console.log(data)

      resolve();
    })
  });
}

async function run() {
  await readFile('file1.txt');
  await readFile('file2.txt');
  await readFile('file3.txt');
}

run();

因为您正在调用 readFile 并以相同的异步序列解析它,所以它同时被调用,这是race condition.

您必须等待回调处理然后解决它(在回调范围内)。


推荐阅读