每種語言都有自己的優(yōu)勢,互相結合起來各取所長程序執(zhí)行起來效率更高或者說哪種實現(xiàn)方式較簡單就用哪個,nodejs是利用子進程來調(diào)用系統(tǒng)命令或者文件,文檔見
http://nodejs.org/api/child_process.html,NodeJS子進程提供了與系統(tǒng)交互的重要接口,其主要API有: 標準輸入、標準輸出及標準錯誤輸出的接口。
NodeJS 子進程提供了與系統(tǒng)交互的重要接口,其主要 API 有:
標準輸入、標準輸出及標準錯誤輸出的接口
child.stdin 獲取標準輸入
child.stdout 獲取標準輸出
child.stderr 獲取標準錯誤輸出
獲取子進程的PID:child.pid
提供生成子進程的方法:child_process.spawn(cmd, args=[], [options])
提供直接執(zhí)行系統(tǒng)命令的方法:child_process.exec(cmd, [options], callback)
提供調(diào)用腳本文件的方法:child_process.execFile(file, [args], [options], [callback])
提供殺死進程的方法:child.kill(signal='SIGTERM')
用實例來感受一下,很有意思的,呵呵~~
1、利用子進程調(diào)用系統(tǒng)命令(獲取系統(tǒng)內(nèi)存使用情況)
新建nodejs文件,名為cmd_spawn.js,代碼如下:
代碼如下:
var spawn = require('child_process').spawn;
free = spawn('free', ['-m']);// 捕獲標準輸出并將其打印到控制臺
free.stdout.on('data', function (data) {
console.log('standard output:\n' + data);
});
// 捕獲標準錯誤輸出并將其打印到控制臺
free.stderr.on('data', function (data) {
console.log('standard error output:\n' + data);
});
// 注冊子進程關閉事件
free.on('exit', function (code, signal) {
console.log('child process eixt ,exit:' + code);
});
下面是運行該腳本和直接運行命令'free -m'的結果,一模一樣:
2、執(zhí)行系統(tǒng)命令(child_process.exec())
這個我還是很常用的,功能感覺比上面的強大那么一點點。比如我很喜歡關注天氣,現(xiàn)在我要curl一下天氣的接口返回json格式的數(shù)據(jù),可能我要對它進行一番操作,這里就打印出來不操作。
新建nodejs文件,名為cmd_exec.js:
代碼如下:
var exec = require('child_process').exec;
var cmdStr = 'curl http://www.weather.com.cn/data/sk/101010100.html';
exec(cmdStr, function(err,stdout,stderr){
if(err) {
console.log('get weather api error:'+stderr);
} else {
/*
這個stdout的內(nèi)容就是上面我curl出來的這個東西:
{"weatherinfo":{"city":"北京","cityid":"101010100","temp":"3","WD":"西北風","WS":"3級","SD":"23%","WSE":"3","time":"21:20","isRadar":"1","Radar":"JC_RADAR_AZ9010_JB","njd":"暫無實況","qy":"1019"}}
*/
var data = JSON.parse(stdout);
console.log(data);
}
});
來感受一下直接curl出來和通過運行腳本的出來的結果是一樣一樣的:
3、調(diào)用傳參數(shù)的shell腳本(child_process.execFile())
這個要先準備個shell腳本,比如我要連到一臺服務器,來修改它的密碼,則我要提供IP,user,new pwd,old pwd,新建shell腳本文件change_password.sh:
代碼如下:
#!/bin/shIP=""
NAME=""
PASSWORD=""
NEWPASSWORD=""
while getopts "H:U:P:N:" arg #選項后面的冒號表示該選項需要參數(shù)
do
case $arg in
H)
IP=$OPTARG
;;
U)
NAME=$OPTARG
;;
P)
PASSWORD=$OPTARG
;;
N)
NEWPASSWORD=$OPTARG
;;
) #當有不認識的選項的時候arg為?
echo "含有未知參數(shù)"
exit 1
;;
esac
done
#先獲取userid
USERID=`/usr/bin/ipmitool -I lanplus -H $IP -U $NAME -P $PASSWORD user list | grep root | awk '{print $1}'`
# echo $USERID
#根據(jù)userid來修改密碼
/usr/bin/ipmitool -I lanplus -H $IP -U $NAME -P $PASSWORD user set password $USERID $NEWPASSWORD
然后我準備個nodejs文件來調(diào)用這個shell腳本,叫file_changepwd.js:
代碼如下:
var callfile = require('child_process');
var ip = '1.1.1.1';
var username = 'test';
var password = 'pwd';
var newpassword = 'newpwd';callfile.execFile('change_password.sh',['-H', ip, '-U', username, '-P', password, '-N', newpassword],null,function (err, stdout, stderr) {
callback(err, stdout, stderr);
});
這里就不方便貼運行結果了,不過我可以用人格保證,它是經(jīng)過測試的。
看過上面的,其實調(diào)用python腳本就沒什么懸念了,本質(zhì)上也就是執(zhí)行命令。
4、調(diào)用python腳本(python腳本本身是傳參數(shù)的)
這里插入一個題外話,下面這段是對python傳參數(shù)的簡單說明一下:
代碼如下:
# -*-coding:utf-8 -*-
'''
需要模塊:sys
參數(shù)個數(shù):len(sys.argv)
腳本名: sys.argv[0]
參數(shù)1: sys.argv[1]
參數(shù)2: sys.argv[2]
'''
import sys
print u"腳本名:", sys.argv[0]
for i in range(1, len(sys.argv)):#這里參數(shù)從1開始
print u"參數(shù)", i, sys.argv[i]
運行結果:
我也來準備一個nodejs文件來調(diào)用這個python腳本(我對py_test.py做了修改,見下面),file_python.js:
代碼如下:
var exec = require('child_process').exec;
var arg1 = 'hello'
var arg2 = 'jzhou'
exec('python py_test.py '+ arg1+' '+arg2+' ',function(error,stdout,stderr){
if(stdout.length >1){
console.log('you offer args:',stdout);
} else {
console.log('you don\'t offer args');
}
if(error) {
console.info('stderr : '+stderr);
}
});py_test.py內(nèi)容如下:
# -*-coding:utf-8 -*-
import sys
print sys.argv
運行結果如下: