node的process以及child_process模块学习笔记


Posted in Javascript onMarch 06, 2018

在死磕进程一个礼拜后,终于把晦涩难懂文档看明白了,准备把自己的理解分享给大家,也希望大家能指出一些意见

进程的概念

  1. 在Node.js中每个应用程序都是一个进程类的实例对象。
  2. 使用process对象代表应用程序,这是一个全局对象,可以通过它来获取Node.jsy应用程序以及运行该程序的用户、环境等各种信息的属性、方法和事件。

进程中几个重要的属性

  1. stdin 标准输入可读流
  2. stdout 标准输入可写流
  3. stderr 标准错误输出流
  4. argv 终端输入参数数组
  5. env 操作系统环境信息
  6. pid 应用程序进程id

stdin以及stdout

process.stdin.on('data', (chunk) => {
 process.stdout.write('进程接收到数据' + chunk)
})

运行结果

node的process以及child_process模块学习笔记

argv

console.log(process.env)

node的process以及child_process模块学习笔记

env: 在mac终端输入 export NODE_ENV=develop

console.log(process.env.NODE_ENV) //develop

进程的方法

  1. process.memoryUsage() 查看内存使用信息
  2. process.nextTick() 当前eventloop执行完毕执行回调函数
  3. process.chdir() chdir方法用于修改Node.js应用程序中使用的当前工作目录
  4. process.cwd() 进程当前工作目录
  5. process.kill() 杀死进程
  6. process.uncaughtException() 当应用程序抛出一个未被捕获的异常时触发进程对象的uncaughtException事件
say() //方法不存在
process.on('uncaughtException',function(err){
 console.log('捕获到一个未被处理的错误:',err);
});

child_process

子进程是今天要讲的重点,我也有一些不太明白,希望能和大家多多交流

child_process出现的背景

在Node.js中,只有一个线程执行所有操作,如果某个操作需要大量消耗CPU资源的情况下,后续操作都需要等待。

在Node.js中,提供了一个child_process模块,通过它可以开启多个子进程,在多个子进程之间可以共享内存空间,可以通过子进程的互相通信来实现信息的交换。

child_process模块给予node任意创建子进程的能力,node官方文档对于child_proces模块给出了四种方法,映射到操作系统其实都是创建子进程。但对于开发者而已,这几种方法的api有点不同

child_process.exec(command[, options][, callback]) 启动
子进程来执行shell命令,可以通过回调参数来获取脚本shell执行结果

child_process.execfile(file[, args][, options][, callback])
与exec类型不同的是,它执行的不是shell命令而是一个可执行文件

child_process.spawn(command[, args][, options])仅仅执行一个shell命令,不需要获取执行结果

child_process.fork(modulePath[, args][, options])可以用node
执行的.js文件,也不需要获取执行结果。fork出来的子进程一定是node进程

spawn

语法:child_process.spawn(command, [args], [options])

  1. command 必须指定的参数,指定需要执行的命令
  2. args 数组,存放了所有运行该命令需要的参数
  3. options 参数为一个对象,用于指定开启子进程时使用的选项
const { spawn } = require('child_process')
const path = require('path')

let child1 = spawn('node', ['test1.js', 'yanyongchao'], {
 stdio: ['pipe', 'pipe', 'pipe'], // 三个元素数组 下面会详解
 cwd: __dirname, 子进程工作目录
 env: process.env, 环境变量
 detached: true // 如果为true,当父进程不存在时也可以独立存在
})

其实上面都好理解除了sdtio数组,下面来一起分析stdio

stdio

stdio是一个数组,用来设置标准输入,标准输出,错误输出。个人理解

pipe:父进程和子进程之间建立一个管道

主进程代码

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

let p = spawn('node', ['childs_t.js'], {
 cwd: path.join(__dirname, 'childs'),
 stdio: ['pipe', 'pipe', process.stderr]
})

p.stdout.on('data', (data) => {
 console.log(data.toString())
}) 
// 这里用stdout原因: 子进程的数据流与常规理解的数据流方向相反,
// stdin:写入流,stdout、stderr:读取流。

子进程代码

process.stdout.write('asd')

如果在stdio中放一个流,process.stdout,process.stdin

主进程代码

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

// 如果放的是一个流,则意味着父进程和子进程共享一个流
const p = spawn('node', ['child_t.js'], {
 cwd: path.join(__dirname, 'childs'),
 stdio: [process.stdin, process.stdout, process.stderr]
})

子进程代码

process.stdout.write('asd') //控制台会输出asd

ipc

主进程代码

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

let p = spawn('node', ['child_t.js'], {
 cwd: path.join(__dirname, 'childs'),
 stdio: ['ipc', 'pipe', 'pipe']
})

p.on('message', (msg) => {
 console.log(msg)
})

p.send('hello chhild_process')

子进程代码

process.on('message', (msg) => {
 process.send('子进程' + msg)
})
// child.send(message,[sendHandle]);//在父进程中向子进程发送消息
// process.send(message,[sendHandle]);//在子进程中向主进程发送消息

detached模式

const { spawn } = require('child_process')
const fs = require('fs')
const path = require('path')
let out = fs.openSync(path.join(__dirname, 'childs/msg.txt'), 'w', 0o666)

let p = spawn('node', ['test4.js'], {
 detached: true, //保证父进程结束,子进程仍然可以运行
 stdio: 'ignore',
 cwd: path.join(__dirname, 'childs')
})

p.unref()

p.on('close', function() {
 console.log('子进程关闭')
})

p.on('exit', function() {
 console.log('子进程退出')
})

p.on('error', function(err) {
 console.log('子进程1开启失败' + err)
})

fork开启一个子进程

  1. 衍生一个新的 Node.js 进程,并通过建立一个 IPC 通讯通道来调用一个指定的模块,该通道允许父进程与子进程之间相互发送信息
  2. fork方法返回一个隐式创建的代表子进程的ChildProcess对象
  3. 子进程的输入/输出操作执行完毕后,子进程不会自动退出,必须使用process.exit()方法显式退出

子进程代码

const { fork } = require('child_process')
const path = require('path')
let child = fork(path.join(__dirname, 'childs/fork1.js'))

child.on('message', (data) => {
 console.log('父进程接收到消息' + data)
})

child.send('hello fork')

child.on('error', (err) => {
 console.error(err)
})

子进程代码

process.on('message', (m, setHandle) => {
 console.log('子进程接收到消息' + m)
 process.send(m) //sendHandle是一个 net.Socket 或 net.Server 对象
})

exec开启子进程

// exec同步执行一个shell命令
let { exec } = require('child_process')
let path = require('path')

// 用于使用shell执行命令, 同步方法

let p1 = exec('node exec.js a b c', {cwd: path.join(__dirname, 'childs')}, function(err, stdout, stderr) {
 console.log(stdout)
})

execFile开启子进程

let { execFile } = require('child_process')
let path = require('path')

let p1 = execFile('node', ['exec.js', 'a', 'b', 'c'], {
 cwd: path.join(__dirname, 'childs')
}, function(err, stdout, stderr) {
 console.log(stdout)
})

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

Javascript 相关文章推荐
利用JQuery的load函数动态加载其它页面的内容的实现代码
Dec 14 Javascript
js前台判断开始时间是否小于结束时间
Feb 23 Javascript
利用谷歌地图API获取点与点的距离的js代码
Oct 11 Javascript
jquery与js函数冲突的两种解决方法
Sep 09 Javascript
Bootstrap图片轮播组件Carousel使用方法详解
Oct 20 Javascript
jquery pagination插件动态分页实例(Bootstrap分页)
Dec 23 Javascript
详解在不使用ssr的情况下解决Vue单页面SEO问题
Nov 08 Javascript
vue router 组件的高级应用实例代码
Apr 08 Javascript
js作用域和作用域链及预解析
Apr 11 Javascript
vue.js表单验证插件(vee-validate)的使用教程详解
May 23 Javascript
vue中利用Promise封装jsonp并调取数据
Jun 18 Javascript
在vue中给后台接口传的值为数组的格式代码
Nov 12 Javascript
Vue2.0实现组件数据的双向绑定问题
Mar 06 #Javascript
轻松搞定jQuery+JSONP跨域请求的解决方案
Mar 06 #jQuery
Vue2.0 http请求以及loading展示实例
Mar 06 #Javascript
浅析Vue中method与computed的区别
Mar 06 #Javascript
浅谈Koa2框架利用CORS完成跨域ajax请求
Mar 06 #Javascript
JavaScript基础心法 深浅拷贝(浅拷贝和深拷贝)
Mar 05 #Javascript
JavaScript基础心法 数据类型
Mar 05 #Javascript
You might like
PHP 代码规范小结
2012/03/08 PHP
openflashchart 2.0 简单案例php版
2012/05/21 PHP
php自定义函数截取汉字长度
2014/05/15 PHP
微信公众平台天气预报功能开发
2014/07/06 PHP
ThinkPHP字符串函数及常用函数汇总
2014/07/18 PHP
PHP实现的比较完善的购物车类
2014/12/02 PHP
PHP数组相加操作及与array_merge的区别浅析
2016/11/26 PHP
Laravel学习笔记之Artisan命令生成自定义模板的方法
2018/11/22 PHP
PHP基于GD2函数库实现验证码功能示例
2019/01/27 PHP
超强的IE背景图片闪烁(抖动)的解决办法
2007/09/09 Javascript
基于jquery实现的树形菜单效果代码
2015/09/06 Javascript
JS实现点击表头表格自动排序(含数字、字符串、日期)
2017/01/22 Javascript
详解express与koa中间件模式对比
2017/08/07 Javascript
老生常谈JavaScript面向对象基础与this指向问题
2017/10/16 Javascript
jQuery 禁止表单用户名、密码自动填充功能
2017/10/30 jQuery
微信小程序实现点击效果
2019/06/21 Javascript
JavaScript:ES2019 的新特性(译)
2019/08/08 Javascript
详解vue-router 动态路由下子页面多页共活的解决方案
2019/12/22 Javascript
JS关闭子窗口并且刷新上一个窗口的实现示例
2020/03/10 Javascript
[55:39]DOTA2-DPC中国联赛 正赛 VG vs LBZS BO3 第二场 1月19日
2021/03/11 DOTA
python批量从es取数据的方法(文档数超过10000)
2018/12/27 Python
python中的split、rsplit、splitlines用法说明
2020/10/23 Python
详解CSS3+JS完美实现放大镜模式
2020/12/03 HTML / CSS
出门问问全球官方商城:Tichome音箱和TicWatch智能手表
2017/12/02 全球购物
美国隐形眼镜网上商店:Lens.com
2019/09/03 全球购物
以太网Ethernet IEEE802.3
2013/08/05 面试题
生产主管岗位职责
2013/11/10 职场文书
运动会广播稿100字
2014/01/11 职场文书
适用于所有创业者的创业计划书
2014/02/05 职场文书
市场营销管理毕业生自荐信
2014/03/03 职场文书
关于感恩的演讲稿400字
2014/08/26 职场文书
2014年人事工作总结范文
2014/11/19 职场文书
导游词之台湾安平古堡
2019/12/25 职场文书
MySQL之高可用集群部署及故障切换实现
2021/04/22 MySQL
MYSQL 无法识别中文的永久解决方法
2021/06/03 MySQL
使用Springboot实现健身房管理系统
2021/07/01 Java/Android