插件窝 干货文章 零开销异步/等待

零开销异步/等待

class error 回调 amp 949    来源:    2024-10-22

node.js 中的回调明显快于 promise。如何在受益于 promise 和 async/await 语法的同时获得回调的性能?

javascript 可以引入 callbackawaitexpression,其语法看起来非常像现有的 awaitexpression,但它会在回调而不是承诺之上运行。#?? #

回调将具有以下形式:


type callback<v e="unknown"> =
  | (error: e) =&gt; void;
  | (error: void, value: v) =&gt; void;
</v>

新的

callbackawaitexpression 将有一个额外的 identifier callback 类型参数,在语法上位于 wait 关键字和正在等待的表达式之间,例如,请注意cb标识符:

await cb fs.readfile('myfile.txt', 'utf-8', cb);

同样,异步函数语法也将扩展为允许

asynccallbackfunction类型。同样,语法将允许单个回调标识符:

async cb function(args, cb) {
  // ...
}

将所有这些放在一起,这将允许编写异步/等待语法驱动的代码,同时受益于回调的性能。它允许编写这样的代码:


async _ function getfiledata(filename, _) {
  try {
    const data = await _ fs.readfile('myfile.txt', 'utf-8', _);
    return 'mydata: ' + data;
  } catch (error) {
    if (!!error &amp;&amp; typeof error === 'object' &amp;&amp; error.code === 'enoent') {
      throw new error('not found');
    }
    throw error;
  }
}

代码相当于现有的javascript:


function getfiledata(filename, callback) {
  const oncatch = (error) =&gt; {
    if (!!error &amp;&amp; typeof error === 'object' &amp;&amp; error.code === 'enoent') {
      callback(new error('not found'));
    } else {
      callback(error);
    }
  };
  try {
    fs.readfile('myfile.txt', 'utf-8', (err, data) =&gt; {
      if (err) {
        oncatch(err);
      } else {
        try {
          callback(null, 'mydata: ' + data);
        } catch (error) {
          oncatch(error);
        }
      }
    });
  } catch (error) {
    oncatch(error);
  }
}

以下是使用现有 async/await 语法的上述代码的样子。该代码几乎等同于

async/await 回调 提案,但由于 promise 的使用,性能较差。

async function getfiledataasync(filename) {
    try {
    const data = await promisify(fs.readfile)('myfile.txt', 'utf-8');
    return 'mydata: ' + data;
  } catch (error) {
    if (!!error &amp;&amp; typeof error === 'object' &amp;&amp; error.code === 'enoent') {
      throw new error('not found');
    }
    throw error;
  }
}

或者,同等地,古老的 promisify 实用程序可以将我们的回调驱动函数转换为这个 promise 驱动函数:


const getfiledataasync = promisify(getfiledata);

糖语法

等待时可以显式指定回调标识符并显式将其用作参数:


await mycallback fs.readfile('a.txt', mycallback)

如果没有使用await标识符mycallback,它会自动插入到函数调用中作为最后一个参数:


await _ fs.readfile('a.txt')

定义异步回调函数时,上面的文字建议显式指定回调标识符,如下所示:


async mycallback function getdata(filename, mycallback) {}

相反,只需使用一次 async 关键字代替某些参数即可减少:


function getdata(filename, async) {}

上述语法修改将示例 getfiledata 函数简化为以下内容:


function getfiledata(filename, async) {
  try {
    const data = await _ fs.readfile('myfile.txt', 'utf-8');
    return 'mydata: ' + data;
  } catch (error) {
    // ...
  }
}

与 typescript 一起使用

从typescript的角度来看,要定义一个新的异步回调函数,只需使用callbacktype即可:


type getfiledata = (filename: string, callback: callback<string>) =&gt; void;
</string>

或者,可以引入异步类型简写,以使其更明确:


type getfiledata = (filename: string, async<string>) =&gt; void;
</string>

也许可以叫回调:


type GetFileData = (filename: string, callback<string>) =&gt; void;
</string>