每種語言都有自己的優勢,互相結合起來各取所長程序執行起來效率更高或者說哪種實現方式較簡單就用哪個,nodejs是利用子進程來調用系統命令或者文件,文檔見http://nodejs.org/api/child_process.html,NodeJS子進程提供了與系統交互的重要接口,其主要API有: 標准輸入、標准輸出及標准錯誤輸出的接口。
NodeJS 子進程提供了與系統交互的重要接口,其主要 API 有:
標准輸入、標准輸出及標准錯誤輸出的接口
child.stdin 獲取標准輸入
child.stdout 獲取標准輸出
child.stderr 獲取標准錯誤輸出
獲取子進程的PID:child.pid
提供生成子進程的方法:child_process.spawn(cmd, args=[], [options])
提供直接執行系統命令的方法:child_process.exec(cmd, [options], callback)
提供調用腳本文件的方法:child_process.execFile(file, [args], [options], [callback])
提供殺死進程的方法:child.kill(signal='SIGTERM')
用實例來感受一下,很有意思的,呵呵~~
1、利用子進程調用系統命令(獲取系統內存使用情況)
新建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、執行系統命令(child_process.exec())
這個我還是很常用的,功能感覺比上面的強大那么一點點。比如我很喜歡關注天氣,現在我要curl一下天氣的接口返回json格式的數據,可能我要對它進行一番操作,這里就打印出來不操作。
新建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的內容就是上面我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、調用傳參數的shell腳本(child_process.execFile())
這個要先准備個shell腳本,比如我要連到一台服務器,來修改它的密碼,則我要提供IP,user,new pwd,old pwd,新建shell腳本文件change_password.sh:
#!/bin/sh IP="" NAME="" PASSWORD="" NEWPASSWORD=""
while getopts "H:U:P:N:" arg #選項后面的冒號表示該選項需要參數 do
case $arg in H) IP=$OPTARG ;; U) NAME=$OPTARG ;; P) PASSWORD=$OPTARG ;; N) NEWPASSWORD=$OPTARG ;; ?) #當有不認識的選項的時候arg為?
echo "含有未知參數" 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 #根據userid來修改密碼 /usr/bin/ipmitool -I lanplus -H $IP -U $NAME -P $PASSWORD user set password $USERID $NEWPASSWORD
然后我准備個nodejs文件來調用這個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); });
這里就不方便貼運行結果了,不過我可以用人格保證,它是經過測試的。
看過上面的,其實調用python腳本就沒什么懸念了,本質上也就是執行命令。
4、調用python腳本(python腳本本身是傳參數的)
這里插入一個題外話,下面這段是對python傳參數的簡單說明一下:
# -*-coding:utf-8 -*-
''' 需要模塊:sys 參數個數:len(sys.argv) 腳本名: sys.argv[0] 參數1: sys.argv[1] 參數2: sys.argv[2] '''
import sys print u"腳本名:", sys.argv[0] for i in range(1, len(sys.argv)):#這里參數從1開始
print u"參數", i, sys.argv[i]
運行結果:
我也來准備一個nodejs文件來調用這個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內容如下: # -*-coding:utf-8 -*-
import sys print sys.argv
運行結果如下:
還是挺贊的,又為2014完成了一篇精致的博客。哈哈~~