浅谈Node.js 子进程与应用场景


Posted in Javascript onJanuary 24, 2018

背景

由于ons(阿里云 RocketMQ 包)基于 C艹 封装而来,不支持单一进程内实例化多个生产者与消费者,为了解决这一问题,使用了 Node.js 子进程。

在使用的过程中碰到的坑

发布:进程管理关闭主进程后,子进程变为操作系统进程(pid 为 1)

几种解决方案

将子进程看做独立运行的进程,记录 pid,发布时进程管理关闭主进程同时关闭子进程

主进程监听关闭事件,主动关闭从属于自己的子进程

子进程种类

  1. spawn:执行命令
  2. exec:执行命令(新建 shell)
  3. execFile:执行文件
  4. fork:执行文件

子进程常用事件

  1. exit
  2. close
  3. error
  4. message

close 与 exit 是有区别的,close 是在数据流关闭时触发的事件,exit 是在子进程退出时触发的事件。因为多个子进程可以共享同一个数据流,所以当某个子进程 exit 时不一定会触发 close 事件,因为这个时候还存在其他子进程在使用数据流。

子进程数据流

  1. stdin
  2. stdout
  3. stderr

因为是以主进程为出发点,所以子进程的数据流与常规理解的数据流方向相反,stdin:写入流,stdout、stderr:读取流。

spawn

spawn(command[, args][, options])

执行一条命令,通过 data 数据流返回各种执行结果。

基础使用

const { spawn } = require('child_process');

const child = spawn('find', [ '.', '-type', 'f' ]);
child.stdout.on('data', (data) => {
  console.log(`child stdout:\n${data}`);
});

child.stderr.on('data', (data) => {
  console.error(`child stderr:\n${data}`);
});

child.on('exit', (code, signal) => {
  console.log(`child process exit with: code $[code], signal: ${signal}`);
});

常用参数

{
  cwd: String,
  env: Object,
  stdio: Array | String,
  detached: Boolean,
  shell: Boolean,
  uid: Number,
  gid: Number
}

重点说明下 detached 属性,detached 设置为 true 是为子进程独立运行做准备。子进程的具体行为与操作系统相关,不同系统表现不同,Windows 系统子进程会拥有自己的控制台窗口,POSIX 系统子进程会成为新进程组与会话负责人。

这个时候子进程还没有完全独立,子进程的运行结果会展示在主进程设置的数据流上,并且主进程退出会影响子进程运行。当 stdio 设置为 ignore 并调用 child.unref(); 子进程开始真正独立运行,主进程可独立退出。

exec

exec(command[, options][, callback])

执行一条命令,通过回调参数返回结果,指令未执行完时会缓存部分结果到系统内存。

const { exec } = require('child_process');

exec('find . -type f | wc -l', (err, stdout, stderr) => {
  if (err) {
    console.error(`exec error: ${err}`);
    return;
  }

  console.log(`Number of files ${stdout}`);
});

两全其美 —— spawn 代替 exec

由于 exec 的结果是一次性返回,在返回前是缓存在内存中的,所以在执行的 shell 命令输出过大时,使用 exec 执行命令的方式就无法按期望完成我们的工作,这个时候可以使用 spawn 代替 exec 执行 shell 命令。

const { spawn } = require('child_process');

const child = spawn('find . -type f | wc -l', {
  stdio: 'inherit',
  shell: true
});

child.stdout.on('data', (data) => {
  console.log(`child stdout:\n${data}`);
});

child.stderr.on('data', (data) => {
  console.error(`child stderr:\n${data}`);
});

child.on('exit', (code, signal) => {
  console.log(`child process exit with: code $[code], signal: ${signal}`);
});

execFile

child_process.execFile(file[, args][, options][, callback])

执行一个文件

与 exec 功能基本相同,不同之处在于执行给定路径的一个脚本文件,并且是直接创建一个新的进程,而不是创建一个 shell 环境再去运行脚本,相对更轻量级更高效。但是在 Windows 系统中如 .cmd 、 .bat 等文件无法直接运行,这是 execFile 就无法工作,可以使用 spawn、exec 代替。

fork

child_process.fork(modulePath[, args][, options])

执行一个 Node.js 文件

// parent.js

const { fork } = require('child_process');

const child = fork('child.js');

child.on('message', (msg) => {
  console.log('Message from child', msg);
});

child.send({ hello: 'world' });
// child.js

process.on('message', (msg) => {
  console.log('Message from parent:', msg);
});

let counter = 0;

setInterval(() => {
  process.send({ counter: counter++ });
}, 3000);

fork 实际是 spawn 的一种特殊形式,固定 spawn Node.js 进程,并且在主子进程间建立了通信通道,让主子进程可以使用 process 模块基于事件进行通信。

子进程使用场景

  1. 计算密集型系统
  2. 前端构建工具利用多核 CPU 并行计算,提升构建效率
  3. 进程管理工具,如:PM2 中部分功能

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Javascript 相关文章推荐
javascript 原型模式实现OOP的再研究
Apr 09 Javascript
使用jquery实现仿百度自动补全特效
Jul 23 Javascript
jQuery往返城市和日期查询实例讲解
Oct 09 Javascript
js简单判断flash是否加载完成的方法
Jun 21 Javascript
Vue Element使用icon图标教程详解(第三方)
Feb 07 Javascript
Vue在页面右上角实现可悬浮/隐藏的系统菜单
May 04 Javascript
vue中轮训器的使用
Jan 27 Javascript
详解Vue.directive 自定义指令
Mar 27 Javascript
koa2 用户注册、登录校验与加盐加密的实现方法
Jul 22 Javascript
layui实现二维码弹窗、并下载到本地的方法
Sep 25 Javascript
原生JS封装拖动验证滑块的实现代码示例
Jun 01 Javascript
vue 路由meta 设置导航隐藏与显示功能的示例代码
Sep 04 Javascript
除Console.log()外更多的Javascript调试命令
Jan 24 #Javascript
深入理解node.js http模块
Jan 24 #Javascript
微信、QQ、微博、Safari中使用js唤起App
Jan 24 #Javascript
基于node打包可执行文件工具_Pkg使用心得分享
Jan 24 #Javascript
Angular整合zTree的示例代码
Jan 24 #Javascript
使用classList来实现两个按钮样式的切换方法
Jan 24 #Javascript
基于vue.js 2.x的虚拟滚动条的示例代码
Jan 23 #Javascript
You might like
PHP调用JAVA的WebService简单实例
2014/03/11 PHP
laravel5环境隐藏index.php后缀(apache)的方法
2019/10/12 PHP
详解CSS样式中的 !important * _ 符号
2021/03/09 HTML / CSS
如何设置iframe高度自适应在跨域情况下的可用方法
2013/09/06 Javascript
jQuery中remove()方法用法实例
2014/12/25 Javascript
javascript设计模式之中介者模式Mediator
2014/12/30 Javascript
javascript 常见功能汇总
2015/06/11 Javascript
js实现获取div坐标的方法
2015/11/16 Javascript
js判断上传文件后缀名是否合法
2016/01/28 Javascript
Node.js实用代码段之获取Buffer对象字节长度
2016/03/17 Javascript
原生ajax处理json格式数据的实例代码
2016/12/25 Javascript
解决nodejs中使用http请求返回值为html时乱码的问题
2017/02/18 NodeJs
利用vue.js插入dom节点的方法
2017/03/15 Javascript
jQuery中each循环的跳出和结束实例
2017/08/16 jQuery
JavaScript中变量、指针和引用功能与操作示例
2018/08/04 Javascript
angular 实时监听input框value值的变化触发函数方法
2018/08/31 Javascript
Vue2.5学习笔记之如何在项目中使用和配置Vue
2018/09/26 Javascript
简化版的vue-router实现思路详解
2018/10/19 Javascript
解决qrcode.js生成二维码时必须定义一个空div的问题
2020/07/09 Javascript
JavaScript 判断浏览器是否是IE
2021/02/19 Javascript
浅析Python中signal包的使用
2015/11/13 Python
使用Python爬虫爬取小红书完完整整的全过程
2021/01/19 Python
HTML5 embed 标签使用方法介绍
2013/08/13 HTML / CSS
使用html5制作loading图的示例
2014/04/14 HTML / CSS
Marriott中国:万豪国际酒店查询预订
2016/09/02 全球购物
精油和天然健康美容产品:Art Naturals
2018/01/27 全球购物
Michael Kors澳大利亚官网:世界知名的奢侈饰品和成衣设计师
2020/02/13 全球购物
早晨薰衣草在线女性精品店:Morning Lavender
2021/01/04 全球购物
艺术系大学生毕业个人自我评价
2013/09/19 职场文书
《一个小村庄的故事》教学反思
2014/04/13 职场文书
2014年党员评议表自我评价
2014/09/27 职场文书
挂职锻炼工作总结2015
2015/05/28 职场文书
Python3.8官网文档之类的基础语法阅读
2021/09/04 Python
HTML基础详解(下)
2021/10/16 HTML / CSS
PYTHON基于Pyecharts绘制常见的直角坐标系图表
2022/04/28 Python
Mysql 文件配置解析介绍
2022/05/06 MySQL