วิธีอ่านไฟล์ด้วย async / await อย่างถูกต้อง?


121

ฉันคิดไม่ออกว่าasync/ awaitทำงานอย่างไร ฉันเข้าใจเล็กน้อย แต่ไม่สามารถใช้งานได้

function loadMonoCounter() {
    fs.readFileSync("monolitic.txt", "binary", async function(err, data) {
       return await new Buffer( data);
  });
}

module.exports.read = function() {
  console.log(loadMonoCounter());
};

ฉันรู้ว่าฉันสามารถใช้ได้readFileSyncแต่ถ้าฉันทำได้ฉันรู้ว่าฉันจะไม่เข้าใจasync/ awaitและฉันจะฝังปัญหา

เป้าหมาย: เรียกloadMonoCounter()และส่งคืนเนื้อหาของไฟล์

ไฟล์นั้นจะเพิ่มขึ้นทุกครั้งที่incrementMonoCounter()เรียก (ทุกครั้งที่โหลดหน้า) ไฟล์นี้มีการถ่ายโอนข้อมูลของบัฟเฟอร์ในไบนารีและถูกเก็บไว้ใน SSD

ไม่ว่าฉันจะทำอะไรฉันได้รับข้อผิดพลาดหรือundefinedในคอนโซล


สิ่งนี้ตอบคำถามของคุณหรือไม่? การใช้ระบบไฟล์ใน node.js ด้วย async / await
KyleMit

คำตอบ:


167

ในการใช้await/ asyncคุณต้องการวิธีการที่คืนคำสัญญา ฟังก์ชัน API หลักจะไม่ทำเช่นนั้นโดยไม่มีตัวห่อเช่นpromisify:

const fs = require('fs');
const util = require('util');

// Convert fs.readFile into Promise version of same    
const readFile = util.promisify(fs.readFile);

function getStuff() {
  return readFile('test');
}

// Can't use `await` outside of an async function so you need to chain
// with then()
getStuff().then(data => {
  console.log(data);
})

ตามบันทึกreadFileSyncไม่รับการโทรกลับจะส่งคืนข้อมูลหรือแสดงข้อยกเว้น คุณไม่ได้รับค่าที่คุณต้องการเนื่องจากฟังก์ชันที่คุณจัดหาถูกละเว้นและคุณไม่ได้บันทึกค่าที่ส่งคืนจริง


3
ขอบคุณฉันไม่รู้ว่าฉันต้องรวม API หลัก คุณสุดยอดมาก
Jeremy Dicaire

4
API หลักกำหนดไว้ล่วงหน้าเกี่ยวกับข้อกำหนด Promise ที่ทันสมัยและการใช้async/ awaitดังนั้นจึงเป็นขั้นตอนที่จำเป็น ข่าวดีpromisifyมักจะทำให้การทำงานเป็นไปอย่างราบรื่น
tadman

1
สิ่งนี้จะจัดการกับความยุ่งเหยิงของการไม่สามารถใช้ประโยชน์จาก async-await กับ FS ได้ตามปกติ ขอบคุณสำหรับสิ่งนี้! คุณช่วยฉันได้ตัน! ไม่มีคำตอบที่ตอบโจทย์ของคุณได้จริงๆ
jacobhobson

3
นอกจากนี้การรอคอยก็ค่อนข้างซ้ำซ้อนเนื่องจากสามารถอนุมานได้ const file = await readFile...; return file;แต่ถ้าคุณไม่ต้องการที่จะรอคอยอย่างชัดเจนมีในตัวอย่างเช่นคุณสามารถทำได้
bigkahunaburger

1
@shijin จนกว่า Node core API จะเปลี่ยนไปเป็นสัญญาซึ่งไม่น่าเป็นไปได้ในตอนนี้ใช่แล้ว อย่างไรก็ตามมี Wrapper NPM ที่ทำเพื่อคุณ
tadman

152

เนื่องจากสัญญา Node v11.0.0 fs พร้อมใช้งานโดยกำเนิดโดยไม่มีpromisify:

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

4
ไม่มี libs เพิ่มเติมสะอาดและเรียบง่าย - ควรเป็นคำตอบที่ดีกว่า
Adam Bubela

2
ณ วันที่ 21 ต.ค. 2019 v12 เป็นเวอร์ชัน LTS ที่ใช้งานอยู่
cbronson

16
import { promises as fs } from "fs";หากคุณต้องการใช้ไวยากรณ์การนำเข้า
tr3online

21

นี่คือคำตอบของ @ Joel เวอร์ชัน TypeScript สามารถใช้งานได้หลังจากโหนด 11.0:

import { promises as fs } from 'fs';

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

18

คุณสามารถตัดคำสั่ง readFile ได้อย่างง่ายดายด้วยสัญญาดังนี้:

async function readFile(path) {
    return new Promise((resolve, reject) => {
      fs.readFile(path, 'utf8', function (err, data) {
        if (err) {
          reject(err);
        }
        resolve(data);
      });
    });
  }

จากนั้นใช้:

await readFile("path/to/file");

ไม่รอใช้ภายในฟังก์ชัน async?
VikasBhat

@VikasBhat ใช่บรรทัดการรอคอยด้านบนจะถูกใช้ภายในฟังก์ชัน async อื่นเนื่องจากข้อมูลจำเพาะต้องการให้เป็นเช่นนั้น
whoshotdk

8

คุณสามารถfs.promisesใช้ได้ตั้งแต่ Node v11.0.0

import fs from 'fs';

const readFile = async filePath => {
  try {
    const data = await fs.promises.readFile(filePath, 'utf8')
    return data
  }
  catch(err) {
    console.log(err)
  }
}

หากคุณต้องการเพียงแค่ใช้คำสัญญาคุณสามารถทำสิ่งต่างๆเช่นconst fs = require('fs').promises
nathanfranke

โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.