在node.js中使用带有async / await的文件系统

时间:2016-11-14 16:52:22

标签: javascript node.js async-await fs

我想在一些文件系统操作中使用async / await。通常async / await工作正常,因为我使用babel-plugin-syntax-async-functions

但是使用这段代码我遇到了names未定义的if情况:

import fs from 'fs';

async function myF() {
  let names;
  try {
    names = await fs.readdir('path/to/dir');
  } catch (e) {
    console.log('e', e);
  }
  if (names === undefined) {
    console.log('undefined');
  } else {
    console.log('First Name', names[0]);
  }
}

myF();

当我将代码重建到回调地狱版本时,一切正常,我得到了文件名。 谢谢你的提示。

11 个答案:

答案 0 :(得分:85)

从节点8.0.0开始,您可以使用:

import fs from 'fs';
import {promisify} from 'util';

const readdir = promisify(fs.readdir);

async function myF() {
  let names;
  try {
    {err, names} = await readdir('path/to/dir');
    if (err) {
        // Handle the error.
    }
  } catch (e) {
    console.log('e', e);
  }
  if (names === undefined) {
    console.log('undefined');
  } else {
    console.log('First Name', names[0]);
  }
}

myF();

请参阅https://nodejs.org/dist/latest-v8.x/docs/api/util.html#util_util_promisify_original

答案 1 :(得分:39)

Node.js 8.0.0

Native async / await

Promisify

从此版本开始,您可以使用 util 库中的本机Node.js函数。

const fs = require('fs')
const { promisify } = require('util')

const readFileAsync = promisify(fs.readFile)
const writeFileAsync = promisify(fs.writeFile)

const run = async () => {
  const res = await readFileAsync('./data.json')
  console.log(res)
}

run()

承诺包装

const fs = require('fs')

const readFile = (path, opts = 'utf8') =>
  new Promise((resolve, reject) => {
    fs.readFile(path, opts, (err, data) => {
      if (err) reject(err)
      else resolve(data)
    })
  })

const writeFile = (path, data, opts = 'utf8') =>
  new Promise((resolve, reject) => {
    fs.writeFile(path, data, opts, (err) => {
      if (err) reject(err)
      else resolve()
    })
  })

module.exports = {
  readFile,
  writeFile
}

...


// in some file, with imported functions above
// in async block
const run = async () => {
  const res = await readFile('./data.json')
  console.log(res)
}

run()

建议

如果您不想重新抛出异常上限,请始终使用try..catch来等待阻止。

答案 2 :(得分:32)

您可能会产生错误的行为,因为File-Api fs.readdir不会返回承诺。它只需要一个回调。如果你想使用async-await语法,你可以'promisify'这个函数:

function readdirAsync(path) {
  return new Promise(function (resolve, reject) {
    fs.readdir(path, function (error, result) {
      if (error) {
        reject(error);
      } else {
        resolve(result);
      }
    });
  });
}

然后改为呼叫:

names = await readdirAsync('path/to/dir');

答案 3 :(得分:12)

本地异步等待样式自版本10起[实验性]

从Node.JS 10.0.0开始,您可以访问已经实现的文件系统方法,并且可以将它们与try catch异常处理一起使用,而不用检查返回值是否包含错误。

该API是实验性,但它非常干净优雅!只需使用.promises对象的fs成员:

import fs from 'fs';
const fsPromises = fs.promises;

async function listDir() {
  try {
    return await fsPromises.readdir('path/to/dir');
  } catch (err) {
    console.error('Error occured while reading directory!', err);
  }
}

listDir();

答案 4 :(得分:10)

您可以使用fs.Promises

const { promises: fs } = require("fs");

async function myF() {
    let names;
    try {
        names = await fs.readdir("path/to/dir");
    } catch (e) {
        console.log("e", e);
    }
    if (names === undefined) {
        console.log("undefined");
    } else {
        console.log("First Name", names[0]);
    }
}

myF();

使用readFile的示例

const { promises: fs } = require("fs");

async function getContent(filePath, encoding = "utf-8") {
    if (!filePath) {
        throw new Error("filePath required");
    }

    return fs.readFile(filePath, { encoding });
}

(async () => {
    const content = await getContent("./package.json");

    console.log(content);
})();

答案 5 :(得分:10)

这是问题的TypeScript版本。在节点11.0之后可用:

import { promises as fs } from 'fs';

async function loadMonoCounter() {
    const data = await fs.readFile('monolitic.txt', 'binary');
    return Buffer.from(data);
}

答案 6 :(得分:5)

这对我有用:

const fsp = require('fs-promise');

(async () => {
  try {
    const names = await fsp.readdir('path/to/dir');
    console.log(names[0]);
  } catch (e) {
    console.log('error: ', e);
  }
})();

启用harmony flag时,此代码适用于没有babel的节点7.6:node --harmony my-script.js。从节点7.7开始,you don't even need this flag

开头包含的fsp库只是fs(和fs-ext)的保护包装。

我现在真的已经离开了这些天你在没有babel的节点中可以做些什么!原生async / await使编写代码非常愉快!

更新2017-06: fs-promise模块已弃用。请使用fs-extra代替相同的API。

答案 7 :(得分:4)

与自定义函数相比,建议使用诸如https://github.com/davetemplin/async-file之类的npm软件包。例如:

import * as fs from 'async-file';

await fs.rename('/tmp/hello', '/tmp/world');
await fs.appendFile('message.txt', 'data to append');
await fs.access('/etc/passd', fs.constants.R_OK | fs.constants.W_OK);

var stats = await fs.stat('/tmp/hello', '/tmp/world');

其他答案已过时

答案 8 :(得分:4)

节点v14.0.0及更高版本

您可以这样做:

no_file_pan.setText(getFileName(mPhotoFile));

就像从import { readdir } from "fs/promises"; 导入

有关更多详细信息,请参见此PR:https://github.com/nodejs/node/pull/31553

答案 9 :(得分:1)

我有这个小帮助模块,可以导出fs个函数的promisified

const fs = require("fs");
const {promisify} = require("util")

module.exports = {
  readdir: promisify(fs.readdir),
  readFile: promisify(fs.readFile),
  writeFile: promisify(fs.writeFile)
  // etc...
};

答案 10 :(得分:0)

您可以使用简单而轻巧的模块https://github.com/nacholibre/nwc-l,它支持异步和同步方法。

注意:该模块由我创建。