七天学会 Node.js 第五章 进程管理

[TOCM]

Node.js 可以感知和控制自身进程的运行环境和状态,也可以创建子进程并与其协同工作,这使得 Node.js 可以把多个程序组合在一起共同完成某项工作,并在其中充当胶水和调度器的作用。本章除了介绍与之相关的 Node.js 内置模块外,还会重点介绍典型的使用场景。

一、开门红


我们已经知道了 Node.js 自带的fs模块比较基础,把一个目录里的所有文件和子目录都拷贝到另一个目录里需要写不少代码。另外我们也知道,终端下的cp命令比较好用,一条cp -r source/* target命令就能搞定目录拷贝。那我们首先看看如何使用 Node.js 调用终端命令来简化目录拷贝,示例代码如下:

  1. var child_process = require('child_process');
  2. var util = require('util');
  3. function copy(source, target, callback) {
  4. child_process.exec(
  5. util.format('cp -r %s/* %s', source, target), callback);
  6. }
  7. copy('a', 'b', function (err) {
  8. // ...
  9. });

从以上代码中可以看到,子进程是异步运行的,通过回调函数返回执行结果。

二、API 走马观花


我们先大致看看 Node.js 提供了哪些和进程管理有关的 API。这里并不逐一介绍每个 API 的使用方法,官方文档已经做得很好了。

1、Process

官方文档:https://nodejs.org/api/process.html

任何一个进程都有启动进程时使用的命令行参数,有标准输入标准输出,有运行权限,有运行环境和运行状态。在 Node.js 中,可以通过process对象感知和控制 Node.js 自身进程的方方面面。另外需要注意的是,process不是内置模块,而是一个全局对象,因此在任何地方都可以直接使用。

2、Child Process

官方文档:https://nodejs.org/api/child_process.html

使用child_process模块可以创建和控制子进程。该模块提供的 API 中最核心的是.spawn,其余 API 都是针对特定使用场景对它的进一步封装,算是一种语法糖。

3、Cluster

官方文档:https://nodejs.org/api/cluster.html

cluster模块是对child_process模块的进一步封装,专用于解决单进程 Node.js Web 服务器无法充分利用多核 CPU 的问题。使用该模块可以简化多进程服务器程序的开发,让每个核上运行一个工作进程,并统一通过主进程监听端口和分发请求。

三、应用场景


和进程管理相关的 API 单独介绍起来比较枯燥,因此这里从一些典型的应用场景出发,分别介绍一些重要 API 的使用方法。

1、如何获取命令行参数

在 Node.js 中可以通过process.argv获取命令行参数。但是比较意外的是,node执行程序路径和主模块文件路径固定占据了argv[0]argv[1]两个位置,而第一个命令行参数从argv[2]开始。为了让argv使用起来更加自然,可以按照以下方式处理。

  1. function main(argv) {
  2. // ...
  3. }
  4. main(process.argv.slice(2));

2、如何退出程序

通常一个程序做完所有事情后就正常退出了,这时程序的退出状态码为0。或者一个程序运行时发生了异常后就挂了,这时程序的退出状态码不等于0。如果我们在代码中捕获了某个异常,但是觉得程序不应该继续运行下去,需要立即退出,并且需要把退出状态码设置为指定数字,比如1,就可以按照以下方式:

  1. try {
  2. // ...
  3. } catch (err) {
  4. // ...
  5. process.exit(1);
  6. }

3、如何控制输入输出

Node.js 程序的标准输入流(stdin)、一个标准输出流(stdout)、一个标准错误流(stderr)分别对应process.stdinprocess.stdoutprocess.stderr,第一个是只读数据流,后边两个是只写数据流,对它们的操作按照对数据流的操作方式即可。例如,console.log可以按照以下方式实现。

  1. function log() {
  2. process.stdout.write(
  3. util.format.apply(util, arguments) + '\n');
  4. }

4、如何降权

在 Linux 系统下,我们知道需要使用 root 权限才能监听 1024 以下端口。但是一旦完成端口监听后,继续让程序运行在 root 权限下存在安全隐患,因此最好能把权限降下来。以下是这样一个例子。

  1. http.createServer(callback).listen(80, function () {
  2. var env = process.env,
  3. uid = parseInt(env['SUDO_UID'] || process.getuid(), 10),
  4. gid = parseInt(env['SUDO_GID'] || process.getgid(), 10);
  5. process.setgid(gid);
  6. process.setuid(uid);
  7. });

上例中有几点需要注意:

  1. 如果是通过sudo获取 root 权限的,运行程序的用户的 UID 和 GID 保存在环境变量SUDO_UIDSUDO_GID里边。如果是通过chmod +s方式获取 root 权限的,运行程序的用户的 UID 和 GID 可直接通过process.getuidprocess.getgid方法获取。

  2. process.setuidprocess.setgid方法只接受number类型的参数。

  3. 降权时必须先降 GID 再降 UID,否则顺序反过来的话就没权限更改程序的 GID 了。

5、如何创建子进程

以下是一个创建 Node.js 子进程的例子。

  1. var child = child_process.spawn('node', [ 'xxx.js' ]);
  2. child.stdout.on('data', function (data) {
  3. console.log('stdout: ' + data);
  4. });
  5. child.stderr.on('data', function (data) {
  6. console.log('stderr: ' + data);
  7. });
  8. child.on('close', function (code) {
  9. console.log('child process exited with code ' + code);
  10. });

上例中使用了.spawn(exec, args, options)方法,该方法支持三个参数。第一个参数是执行文件路径,可以是执行文件的相对或绝对路径,也可以是根据 PATH 环境变量能找到的执行文件名。第二个参数中,数组中的每个成员都按顺序对应一个命令行参数。第三个参数可选,用于配置子进程的执行环境与行为。

另外,上例中虽然通过子进程对象的.stdout.stderr访问子进程的输出,但通过options.stdio字段的不同配置,可以将子进程的输入输出重定向到任何数据流上,或者让子进程共享父进程的标准输入输出流,或者直接忽略子进程的输入输出。

6、进程间如何通讯

在 Linux 系统下,进程之间可以通过信号互相通信。以下是一个例子。

  1. /* parent.js */
  2. var child = child_process.spawn('node', [ 'child.js' ]);
  3. child.kill('SIGTERM');
  4. /* child.js */
  5. process.on('SIGTERM', function () {
  6. cleanUp();
  7. process.exit(0);
  8. });

在上例中,父进程通过.kill方法向子进程发送SIGTERM信号,子进程监听process对象的SIGTERM事件响应信号。不要被.kill方法的名称迷惑了,该方法本质上是用来给进程发送信号的,进程收到信号后具体要做啥,完全取决于信号的种类和进程自身的代码。

另外,如果父子进程都是 Node.js 进程,就可以通过 IPC(进程间通讯)双向传递数据。以下是一个例子。

  1. /* parent.js */
  2. var child = child_process.spawn('node', [ 'child.js' ], {
  3. stdio: [ 0, 1, 2, 'ipc' ]
  4. });
  5. child.on('message', function (msg) {
  6. console.log(msg);
  7. });
  8. child.send({ hello: 'hello' });
  9. /* child.js */
  10. process.on('message', function (msg) {
  11. msg.hello = msg.hello.toUpperCase();
  12. process.send(msg);
  13. });

可以看到,父进程在创建子进程时,在options.stdio字段中通过ipc开启了一条 IPC 通道,之后就可以监听子进程对象的message事件接收来自子进程的消息,并通过.send方法给子进程发送消息。在子进程这边,可以在process对象上监听message事件接收来自父进程的消息,并通过.send方法向父进程发送消息。数据在传递过程中,会先在发送端使用JSON.stringify方法序列化,再在接收端使用JSON.parse方法反序列化。

7、如何守护子进程

守护进程一般用于监控工作进程的运行状态,在工作进程不正常退出时重启工作进程,保障工作进程不间断运行。以下是一种实现方式。

  1. /* daemon.js */
  2. function spawn(mainModule) {
  3. var worker = child_process.spawn('node', [ mainModule ]);
  4. worker.on('exit', function (code) {
  5. if (code !== 0) {
  6. spawn(mainModule);
  7. }
  8. });
  9. }
  10. spawn('worker.js');

可以看到,工作进程非正常退出时,守护进程立即重启工作进程。

四、小结


本章介绍了使用 Node.js 管理进程时需要的 API 以及主要的应用场景,总结起来有以下几点:

  • 使用process对象管理自身。

  • 使用child_process模块创建和管理子进程。

(完)