1.child_process是Node.js的一个十分重要的模块,通过它可以实现创建多进程,以利用多核计算资源。
child_process
模块提供了四个创建子进程的函数,分别是spawn
,exec
,execFile
和fork
。其中spawn
是最原始的创建子进程的函数,其他三个都是对spawn
不同程度的封装。
spawn
只能运行指定的程序,参数需要在列表中给出,相当于execvp
系统函数,而exec
可以直接运行复杂的命令。
child_process.spawn(command, [args], [options])
child_process.exec(command, [options], callback)
例如要运行ls -lh /usr
,使用spawn
需要写成spawn('ls', ['-lh', '/usr'])
,而exec
只需exec('ls -lh /usr')
。
exec
的实现原理是启动了一个系统shell来解析参数,因此可以是非常复杂的命令,包括管道和重定向。
此外,exec
还可以直接接受一个回调函数作为参数,回调函数有三个参数,分别是err
, stdout
, stderr
,非常方便直接使用,例如:
require('child_process').exec( 'ls -lh /usr' , function(err, stdout , stderr ) {
console.log( stdout );
});
如果使用spawn
,则必须写成:
2.fork
函数用于直接运行Node.js模块,例如fork('./child.js')
,相当于spawn('node', ['./child.js'])
。
与默认的spawn
不同的是,fork
会在父进程与子进程直接建立一个IPC管道,用于父子进程之间的通信。例如
var n = require('child_process').fork( './child.js');
n. on ( 'message', function(m) {
console. log ( 'PARENT got message:', m);
});
n.send({ hello: 'world' });
child.js的内容
process. on ( 'message', function(m) {
console. log ( 'CHILD got message:', m);
});
process.send({ foo: 'bar' });
结果:
PARENT got message: { foo: 'bar' }
CHILD got message: { hello: 'world' }
3.fork
函数有一个问题,就是它只能运行JavaScript代码,如果你喜欢用CoffeeScript(或者其他任何编译到js的语言),是无法通过fork
调用的。
一个简单的方法是把代码编译到JavaScript再运行,但是很不方便,有没有什么办法呢?答案是可以的,
child_process.spawn(command, [args], [options])
通过把options
参数的stdio
设为['ipc']
,
即可在父子进程之间建立IPC管道。例如子进程使用CoffeeScript:
child_process = require ('child_process');
options ={stdio: ['ipc'] };
3 child = child_process.spawn('coffee', ['./child.coffee'], options);
其中只要把spawn
的第一个参数设置为运行对应脚本的解释器,即可运行,例如使用Continuation.js,
只需child = child_process.spawn('continuation', ['./child.coffee'], options)
。