node Promise 转换

2021/07/06

如果你曾经使用 Javascript,你必须有经验的回调。Javascript 以其内置的异步哲学而闻名。从标准内置模块到其他框架和库,回调无处不在。这就是在 Javascript 中完成异步编程。但是回调有一个黑暗的一面。这叫 Callback Hells。

回调地狱的解决方案是 Promises。而不是通过回调同步返回值或传递结果,我们可以返回一个 Promise,我们将来会返回一些值或错误。 有许多像好的库如 Bluebird 和 Q 可以用来处理 Promise。但是内置的模块和很多其他库不能使用 Promises。这些库能够将传统的回调方法转换为 Promise,在 Node v8 中我们现在可以使用 util.promisify 来处理回调方法。根据 Node.js 的文档: util.promisify(original)

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

const stat = util.promisify(fs.stat);
stat(".")
  .then((stats) => {
    // Do something with `stats`
  })
  .catch((error) => {
    // Handle the error.
  });

或者;

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

const stat = util.promisify(fs.stat);

async function callStat() {
  const stats = await stat(".");
  console.log(`This directory is owned by ${stats.uid}`);
}

看传统的例子:

const fs = require('node:fs');
fs.readFile('./index.js', 'utf8', (err, text) => {
  if (err) {
    console.log('Error', err);
  } else {
    console.log(text);
  }
});

现在这个例子可以用 Promises 写成如下:

const fs = require('node:fs');
const util = require('node:util');
const readFile = util.promisify(fs.readFile);
readFile('./index.js', 'utf8')
  .then((text) => {
    console.log(text);
  })
  .catch((err) => {
    console.log('Error', err);
  });

在这里我们习惯于 util.promisify 将 fs.readFile 转换为基于 promise 的方法。现在,而不是采取回调,readFile 方法作为一个 pormise。promise 有两个方法,then 和 catch。then 如果函数调用成功,则用于获取结果。而 catch 用于捕获错误。Promise 的优点在于,我们可以使用异步的链式写法而不会陷入回调地狱

结合 async 和 await 关键字,promise 可以更容易处理。链式太长的话也会使程序难于阅读。async 和 await 使它看起来更清洁。以下是 Async&Await 的上面的例子:

const fs = require('node:fs');
const util = require('node:util');
const readFile = util.promisify(fs.readFile);
(async function () {
  try {
    const text = await readFile('./index.js', 'utf8');
    console.log(text);
  } catch (err) {
    console.log('Error', err);
  }
})();

异步函数改为回调

回调是异步执行的,并且具有有限的堆栈跟踪。如果回调抛出,则进程将发出一个’uncaughtException’ 事件,如果未处理则将退出

util.callbackify(original)
original <Function> An async function
Returns: <Function> a callback style function

const util = require('util');

async function fn() {
  return 'hello world';
}
const callbackFunction = util.callbackify(fn);

callbackFunction((err, ret) => {
  if (err) throw err;
  console.log(ret);
});