博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
[译]理解 Node.js 事件驱动架构
阅读量:6592 次
发布时间:2019-06-24

本文共 10362 字,大约阅读时间需要 34 分钟。

原文地址:

大部分 Node 模块,例如 http 和 stream,都是基于EventEmitter模块实现的,所以它们拥有触发监听事件的能力。

const EventEmitter = require('events');复制代码

事件驱动的世界中,对于大部分 Node.js 函数,通过回调的形式就是最简单的,例如fs.readFile。在这个例子中,事件会被触发一次(当 Node 已经准备好去调用回调函数时),并且回调函数将作为事件处理函数。

首先让我们看一下基本形式。

Node,当你准备好的时候 call 我

Node 控制异步事件最初的形式是通过回调函数。那是在很久以前,那时候 Javascript 还没有支持原生的 Promise 和 async/await 特性。

回调函数最初只是你传递到其他函数的函数。因为 Javascript 中,函数是第一类对象,所以才让这种行为成为可能。

回调函数不代表代码就是异步调用的,理解这一点是非常重要的。一个函数调用回调函数时,既可以通过同步,也可以通过异步。

例如,下面的fileSize函数接受cb作为回调函数,并且可以根据条件,通过异步或同步触发回调。

function fileSize(fileName, cb) {  if (typeof fileName !== 'string') {    return cb(new TypeError('argument should be string')); // 同步  }  fs.stat(fileName, (err, stats) => {    if (err) {      return cb(err); // 异步    }    cb(null, stats.size); // 异步  });}复制代码

注意:这是一个可能会导致意料之外错误的坏实践。设计函数时,回调函数调用最好只通过异步,或者只通过同步。

让我们看一个用回调形式编写,典型异步 Node 函数的简单例子:

const readFileAsArray = function(file, cb) {  fs.readFile(file, function(err, data) {    if (err) {      return cb(err);    }    const lines = data      .toString()      .trim()      .split('\n');    cb(null, lines);  });};复制代码

readFileAsArray参数包括一个路径和一个回调函数。该宿主函数读取文件内容,并将它们分离到 lines 数组中,并将 lines 传入回调函数中。

下面是一个使用案例。假如在同一目录下,我们有一个文件numbers.txt,内容如下:

101112131415复制代码

如果需要找出文件内奇数的数量,我们可以使用readFileAsArray简化代码:

readFileAsArray('./numbers.txt', (err, lines) => {  if (err) throw err;  const numbers = lines.map(Number);  const oddNumbers = numbers.filter(n => n % 2 === 1);  console.log('奇数的数量为:', oddNumbers.length);});复制代码

上面的代码会读取数字内容并转化为字符串数组,将它们解析为数字,并找出奇数。

这里只用了 Node 的回调函数形式。回调函数第一个参数是err错误对象,没有错误时,返回null。宿主函数中,回调函数作为最后一个参数传入其中。在你的函数中,你应该总是这么做。也就是将宿主函数的最后一个参数设置为回调函数,并且将回调函数第一个参数设置为错误对象。

现代 Javascript 对于回调函数的替代方式

现代的 Javascript 中,我们拥有 Promise 对象。Promise 成为异步 API 中回调函数的替代方案。在 Promise 中,是通过一个 Promise 对象来单独处理成功和失败的情况,并且允许我们异步链式调用它们。而不是通过传入回调函数作为参数,并且错误处理也不会在同一个地方。

如果函数readFileAsArray支持 Promise,我们就可以这样使用:

readFileAsArray('./numbers.txt')  .then(lines => {    const numbers = lines.map(Number);    const oddNumbers = numbers.filter(n => n % 2 === 1);    console.log('Odd numbers count:', oddNumbers.length);  })  .catch(console.error);复制代码

我们通过在宿主函数的返回值上,调用函数.then,而不是传入回调函数。函数.then会给我们获取相同行数的数组的途径,就像回调函数版本的一样,并且我们可以像之前一样进行处理。如果想要进行错误处理,我们需要在返回值上调用.catch函数,这让我们在错误发生的时候可以进行处理。

因为在现代 Javascript 中有 Promise 对象,所以让宿主函数支持 Promise 接口变得非常容易。下面是readFileAsArray函数,在已经拥有回调函数接口的情况下,修改成支持 Promise 接口的例子:

const readFileAsArray = function(file, cb = () => {}) {  return new Promise((resolve, reject) => {    fs.readFile(file, function(err, data) {      if (err) {        reject(err);        return cb(err);      }      const lines = data        .toString()        .trim()        .split('\n');      resolve(lines);      cb(null, lines);    });  });};复制代码

我们让函数返回了一个包裹fs.readFile异步调用的 Promise 对象。这个 Promise 对象暴露了两个参数,分别是resolve函数和reject函数。

我们可以使用Promise的reject方法处理错误时的调用。也可以通过resolve函数,处理正常获取数据的调用。

在 Promise 已经被使用的情况下,我们需要做的事情只有为回调函数添加一个默认值。我们可以在参数中使用一个简单,默认的空函数:() => {}

通过 async/await 使用 Promise

当需要循环一个异步函数时,添加 Promise 接口让你的代码运行起来更简单。如果使用回调函数,会变得很杂乱。

Promise 让事情变得简单,而 Generator(生成器)让事情变得更简单了。也就是说,更近代的运行异步代码的方式,是通过使用async函数,这让我们可以使用同步的方式书写异步代码,也让代码可读性更强。

下面是通过 async/await 的方式,告诉了我们该如何使用readFileAsArray函数的例子:

async function countOdd() {  try {    const lines = await readFileAsArray('./numbers');    const numbers = lines.map(Number);    const oddCount = numbers.filter(n => n % 2 === 1).length;    console.log('Odd numbers count:', oddCount);  } catch (err) {    console.error(err);  }}countOdd();复制代码

首先,我们创建了一个异步函数,只是比正常函数的前面多了一个async字段。在这个异步函数中,我们通过await关键字,调用readFileAsArray函数,就像这个函数直接返回了行数一样。然后,调用readFileAsArray的代码就像同步一样。

我们执行异步函数,让它可以运作。这非常简单并且更具可读性。如果想要进行错误处理,我们需要把异步调用包裹在try/catch语句中。

通过 async/await 特性,我们不需要使用一些特殊的 API(例如.then 和.catch)。我们只需要标记函数,并使用原生的 Javascript 代码就可以了。

只要函数支持 Promise 接口,我们就可以使用 async/await 特性。但是,在 async 函数中,我们不能使用回调函数形式的代码(例如 setTimeout)。

EventEmitter 模块

在 Node 中,EventEmitter 是一个可以加快对象之间通信的模块,也是 Node 异步事件驱动架构的核心。许多 Node 内建模块也是继承于 EventEmitter 的。

核心概念非常简单:Emitter 对象触发具名事件,这会导致事先注册了监听器的具名事件被调用。所以,一个 Emitter 对象拥有两个基本特性:

  • 触发事件
  • 注册和取消注册监听函数

我们只需要创建一个继承于 EventEmitter 的类,就可以让 EventEmitter 起作用了。

class MyEmitter extends EventEmitter {  //}复制代码

Emitter 对象是基于 EventEmitter 类的实例化对象:

const myEmitter = new MyEmitter();复制代码

在 Emitter 对象生命周期的任何时刻,我们都可以通过使用 emit 函数去触发我们想要的具名事件。

myEmitter.emit('something-happened');复制代码

触发事件是某些条件发生了的标志。这个条件通常是 Emitter 对象中状态的变化产生的。

我们通过使用方法on添加监听函数。每当 Emitter 对象触发相关联的事件时,这些函数将会被调用。

事件 !== 异步

让我们看一个例子:

const EventEmitter = require('events');class WithLog extends EventEmitter {  execute(taskFunc) {    console.log('Before executing');    this.emit('begin');    taskFunc();    this.emit('end');    console.log('After executing');  }}const withLog = new WithLog();withLog.on('begin', () => console.log('About to execute'));withLog.on('end', () => console.log('Done with execute'));withLog.execute(() => console.log('*** Executing task ***'));复制代码

WithLog类是一个事件 Emitter。它定义了实例属性execute。这个excute函数接收一个参数,也就是一个任务函数,并把这个函数包裹在 log 语句中。它在执行前后触发了事件。

为了能够看到执行的先后顺序,我们注册了两个事件,并通过一个任务去触发它们。

下面代码的输出结果:

Before executingAbout to execute*** Executing task ***Done with executeAfter executing复制代码

关于上面这个输出信息,我想让你注意的就是所有代码是同步进行的,而不是通过异步。

  • 首先执行 "Before executing" 这一行。
  • begin事件触发执行 "About to execute" 这一行。
  • 实际执行输出 *** Executing task ***
  • end事件触发执行 "Done with execute" 这一行。
  • 最后我们得到 "After executing"

就像老式的回调函数一样,所以千万不要认为事件就意味着代码是同步的或者是异步的。

这个概念很重要,因为如果我们传入一个异步taskFunc来进行execute,事件触发顺序就不再精确。

我们可以通过setImmediate模拟这种情况:

// ...withLog.execute(() => {  setImmediate(() => {    console.log('*** Executing task ***');  });});复制代码

下面是输出结果:

Before executingAbout to executeDone with executeAfter executing*** Executing task ***复制代码

这样是错误的。如果使用了异步调用,将会在调用了Done with executeAfter executing之后,才会执行这一行代码,这样将不再精确。

为了在异步函数调用完成之后触发事件,我们需要通过基于事件的通信,绑定回调函数(或者是 Promise)。下面这个例子做了示范。

使用事件,而不使用回调的一个好处就是我们可以通过注册多个监听器,对相同信号的事件进行多次响应。如果通过回调完成相同的事情,我们必须在单个回调中写更多的逻辑代码。对于应用程序,事件系统是一个在应用顶级构建功能的极好方式,这也允许我们扩展多个插件。你也可以认为是一个状态变化后,允许我们自定义任务的钩子点。

异步事件

让我们把刚才同步的例子转化为异步,这样可以让代码更实用一些。

const fs = require('fs');const EventEmitter = require('events');class WithTime extends EventEmitter {  execute(asyncFunc, ...args) {    this.emit('begin');    console.time('execute');    asyncFunc(...args, (err, data) => {      if (err) {        return this.emit('error', err);      }      this.emit('data', data);      console.timeEnd('execute');      this.emit('end');    });  }}const withTime = new WithTime();withTime.on('begin', () => console.log('About to execute'));withTime.on('end', () => console.log('Done with execute'));withTime.execute(fs.readFile, __filename);复制代码

WithTime类执行一个asyncFunc函数,并通过使用console.timeconsole.timeEnd打印asyncFunc运行的时间。它触发了事件执行前后,正确的顺序。并且也使用异步调用常规的标志,去触发error/data事件。

我们通过调用异步函数fs.readFile测试withTime。我们现在可以通过监听 data 事件,而不必使用回调来处理文件数据。

当执行这些代码时,我们如期地获取到了正确顺序,并且获取了代码执行所用的事件,这非常有用:

About to executeexecute: 4.507msDone with execute复制代码

那我们该如何做才能将回调函数和事件触发器结合起来呢?如果asyncFunc也支持 Promise,我们可以使用 async/await 特性完成同样的事情:

class WithTime extends EventEmitter {  async execute(asyncFunc, ...args) {    this.emit('begin');    try {      console.time('execute');      const data = await asyncFunc(...args);      this.emit('data', data);      console.timeEnd('execute');      this.emit('end');    } catch (err) {      this.emit('error', err);    }  }}复制代码

总之,这种方式的代码对我来说比回调函数和.then/.catch 的方式更具可读性。async/await 特性让我们更贴近 Javascript 语言本身,这无疑是一大成功。

事件参数和错误

在上面的例子中,两个事件被触发的时候,都附带了额外的参数。

error 事件触发时,附带了错误对象。

this.emit('error', err);复制代码

data 事件触发时,附带了 data 数据。

this.emit('data', data);复制代码

我们可以在具名事件中附带很多参数,所有的这些参数可以在之前注册的监听器函数中访问到。

例如,data 事件可用时,我们注册的监听函数就可以获取到事件触发时传递的参数。这个 data 对象就是asyncFunc暴露的。

withTime.on('data', data => {  // do something with data});复制代码

通常error事件是比较特殊的一个。在基于回调函数的例子中,如果我们没有设置错误事件的监听器,node 进程将会自动退出。

为了示范,添加了另外一个执行错误参数方法的回调:

class WithTime extends EventEmitter {  execute(asyncFunc, ...args) {    console.time('execute');    asyncFunc(...args, (err, data) => {      if (err) {        return this.emit('error', err); // Not Handled      }      console.timeEnd('execute');    });  }}const withTime = new WithTime();withTime.execute(fs.readFile, ''); // BAD CALLwithTime.execute(fs.readFile, __filename);复制代码

上面的第一个 execute 调用将会引发错误。node 进程将会崩溃并退出:

events.js:163      throw er; // Unhandled 'error' event      ^Error: ENOENT: no such file or directory, open ''复制代码

而第二个 execute 调用会因为程序崩溃受到影响,并且永远不会执行。

如果我们注册了一个特殊的error事件,node 进程的行为将会改变。例如:

withTime.on('error', err => {  // do something with err, for example log it somewhere  console.log(err);});复制代码

如果我们像上面这样做,来自第一个 execute 调用的错误将会被报告给事件,从而 node 进程就不会崩溃和退出了。另外一个 execute 调用将会正常执行:

{ Error: ENOENT: no such file or directory, open '' errno: -2, code: 'ENOENT', syscall: 'open', path: '' }execute: 4.276ms复制代码

注意现在基于 promise 的 Node 的行为将有所不同,只是会输出一个警告,但是最终将会改变。

UnhandledPromiseRejectionWarning: Unhandled promise rejection (rejection id: 1): Error: ENOENT: no such file or directory, open ''DeprecationWarning: Unhandled promise rejections are deprecated. In the future, promise rejections that are not handled will terminate the Node.js process with a non-zero exit code.复制代码

另外一个捕获错误事件的方式是通过注册一个全局uncaughtException事件。然而,通过这个事件全局捕获错误不是一个好主意。

避免使用uncaughtException,但是如果你必须使用它(比如报告发生了什么或者做清除),你应该让你的程序无论如何都要退出:

process.on('uncaughtException', err => {  // something went unhandled.  // Do any cleanup and exit anyway!  console.error(err); // don't do just that.  // FORCE exit the process too.  process.exit(1);});复制代码

然而,想象一下,如果许多错误事件在同一个时间触发。这意味着上面的uncaughtException监听函数将会触发很多次,这对于一些清除代码可能会发生问题。比如当许多数据库调用发生时,就停止操作。

EventEmitter模块暴露了一个once方法。这个方法意味着只会调用监听器一次,而不是每一次事件触发都调用。所以,这是一个uncaughtException的实际用例,因为发生了第一个未捕获的异常时,我们将开始清除,而且无论如何进程都将会退出。

监听器的顺序

如果我们在同一个事件上,注册了多个监听器,这些监听器的调用将按照顺序进行。也就是说,第一个注册的监听函数,将会被第一个调用。

// 第一个withTime.on('data', data => {  console.log(`Length: ${data.length}`);});// 另一个withTime.on('data', data => {  console.log(`Characters: ${data.toString().length}`);});withTime.execute(fs.readFile, __filename);复制代码

上面的代码将会先执行 Length 这一行,后执行 Characters 这一行,因为这是按照我们定义监听器的顺序执行的。

如果你需要定义一个新的监听器,但是如果需要将这个监听器设置为第一个被调用,你需要使用prependListener方法:

// 第一个withTime.on('data', data => {  console.log(`Length: ${data.length}`);});// 另一个withTime.prependListener('data', data => {  console.log(`Characters: ${data.toString().length}`);});withTime.execute(fs.readFile, __filename);复制代码

这面的代码将会让 Characters 先被打印。

最后,如果你需要删除某一个监听器,你可以使用removeListener方法。

这就是本次话题的所有内容。感谢你的阅读!期待下一次!

转载于:https://juejin.im/post/5b8c97bee51d4538db34d7ba

你可能感兴趣的文章
Android Content Provider的启动过程源码分析
查看>>
POJ1274 The Perfect Stall【二部图最大匹配】
查看>>
极速理解设计模式系列:15.中介者模式(Mediator Pattern)
查看>>
Netty 实现HTTP文件服务器
查看>>
【MFC两种视频图像採集方法】DirectShow与Opencv
查看>>
ORACLE数据库表解锁record is locked by another user
查看>>
MapGuide应用最佳实践--采用托管(Managed)资源还是非托管(Unmanaged)资源
查看>>
iOS:扩展UIColor,支持十六进制颜色设置
查看>>
MapGuide开发中使用Fusion Viewer及通过程序开关图层
查看>>
最简单的基于FFmpeg的AVDevice例子(读取摄像头)【转】
查看>>
解决eclipse不识别Android手机的问题
查看>>
标准答案-有多少人能分得清值类型和引用类型?
查看>>
Java8-Stream-No.12
查看>>
Java编译那些事儿【转】
查看>>
各种排序算法的总结
查看>>
[.net 面向对象程序设计进阶] (25) 团队开发利器(四)分布式版本控制系统Git——使用GitStack+TortoiseGit 图形界面搭建Git环境【转】...
查看>>
SpringBoot相关
查看>>
[LeetCode] Sudoku Solver 求解数独
查看>>
html5/haXe开发偶感
查看>>
js深入研究之神奇的匿名函数类生成方式
查看>>