當我們須要調用系統的命令的時候,最先考慮的os模塊。用os.system()和os.popen()來進行操作。可是這兩個命令過於簡單,不能完畢一些復雜的操作,如給執行的命令提供輸入或者讀取命令的輸出,推斷該命令的執行狀態,管理多個命令的並行等等。這時subprocess中的Popen命令就能有效的完畢我們須要的操作。在這里對Popen予以簡介。
以下是一個非常easy的樣例,來自Python的官網教程:http://docs.python.org/library/subprocess.html
>>> import shlex, subprocess >>> command_line = raw_input() /bin/vikings -input eggs.txt -output "spam spam.txt" -cmd "echo '$MONEY'" >>> args = shlex.split(command_line) >>> print args ['/bin/vikings', '-input', 'eggs.txt', '-output', 'spam spam.txt', '-cmd', "echo '$MONEY'"] >>> p = subprocess.Popen(args) # Success!
Popen它的構造函數例如以下:
subprocess.Popen(args, bufsize=0, executable=None, stdin=None, stdout=None, stderr=None, preexec_fn=None, close_fds=False, shell=False, cwd=None, env=None, universal_newlines=False, startupinfo=None, creationflags=0)
參數args能夠是字符串或者序列類型(如:list,元組),用於指定進程的可運行文件及其參數。假設是序列類型,第一個元素一般是可運行文件的路 徑。我們也能夠顯式的使用executeable參數來指定可運行文件的路徑。在windows操作系統上,Popen通過調用 CreateProcess()來創建子進程,CreateProcess接收一個字符串參數,假設args是序列類型,系統將會通過 list2cmdline()函數將序列類型轉換為字符串。
參數bufsize:指定緩沖。我到如今還不清楚這個參數的詳細含義,望各個大牛指點。
參數executable用於指定可執行程序。普通情況下我們通過args參數來設置所要執行的程序。假設將參數shell設為 True,executable將指定程序使用的shell。在windows平台下,默認的shell由COMSPEC環境變量來指定。
參數stdin, stdout, stderr分別表示程序的標准輸入、輸出、錯誤句柄。他們能夠是PIPE,文件描寫敘述符或文件對象,也能夠設置為None,表示從父進程繼承。
參數preexec_fn僅僅在Unix平台下有效,用於指定一個可執行對象(callable object),它將在子進程執行之前被調用。
參數Close_sfs:在windows平台下,假設close_fds被設置為True,則新創建的子進程將不會繼承父進程的輸入、輸出、錯誤管 道。我們不能將close_fds設置為True同一時候重定向子進程的標准輸入、輸出與錯誤(stdin, stdout, stderr)。
假設參數shell設為true,程序將通過shell來運行。
參數cwd用於設置子進程的當前文件夾。
參數env是字典類型,用於指定子進程的環境變量。假設env = None,子進程的環境變量將從父進程中繼承。
參數Universal_newlines:不同操作系統下,文本的換行符是不一樣的。如:windows下用’/r/n’表示換,而Linux下用 ‘/n’。假設將此參數設置為True,Python統一把這些換行符當作’/n’來處理。
參數startupinfo與createionflags僅僅在windows下用效,它們將被傳遞給底層的CreateProcess()函數,用 於設置子進程的一些屬性,如:主窗體的外觀,進程的優先級等等。
subprocess.PIPE
在創建Popen對象時,subprocess.PIPE能夠初始化stdin, stdout或stderr參數。表示與子進程通信的標准流。
subprocess.STDOUT
創建Popen對象時,用於初始化stderr參數,表示將錯誤通過標准輸出流輸出。
Popen的方法:
Popen.poll()
用於檢查子進程是否已經結束。設置並返回returncode屬性。
Popen.wait()
等待子進程結束。設置並返回returncode屬性。
Popen.communicate(input=None)
與子進程進行交互。向stdin發送數據,或從stdout和stderr中讀取數據。可選參數input指定發送到子進程的參數。 Communicate()返回一個元組:(stdoutdata, stderrdata)。注意:假設希望通過進程的stdin向其發送數據,在創建Popen對象的時候,參數stdin必須被設置為PIPE。相同,如 果希望從stdout和stderr獲取數據,必須將stdout和stderr設置為PIPE。
Popen.send_signal(signal)
向子進程發送信號。
Popen.terminate()
停止(stop)子進程。在windows平台下,該方法將調用Windows API TerminateProcess()來結束子進程。
Popen.kill()
殺死子進程。
Popen.stdin
假設在創建Popen對象是,參數stdin被設置為PIPE,Popen.stdin將返回一個文件對象用於策子進程發送指令。否則返回None。
Popen.stdout
假設在創建Popen對象是,參數stdout被設置為PIPE,Popen.stdout將返回一個文件對象用於策子進程發送指令。否則返回 None。
Popen.stderr
假設在創建Popen對象是,參數stdout被設置為PIPE,Popen.stdout將返回一個文件對象用於策子進程發送指令。否則返回 None。
Popen.pid
獲取子進程的進程ID。
Popen.returncode
獲取進程的返回值。假設進程還沒有結束,返回None。
以下是一個很easy的樣例,來演示supprocess模塊怎樣與一個控件台應用程序進行交 互。
import subprocess
p = subprocess.Popen(“app2.exe”, stdin = subprocess.PIPE, /
stdout = subprocess.PIPE, stderr = subprocess.PIPE, shell = False)
p.stdin.write(’3/n’)
p.stdin.write(’4/n’)
print p.stdout.read()
#—- 結果 —-
input x:
input y:
3 + 4 = 7
app2.exe也是一個很easy的控制台程序,它從界面上接收兩個數值,運行加操作,並將結 果打印到控制台上。代碼例如以下:
#include <iostream>
using namespace std;
int main(int argc, const char *artv[])
{
int x, y;
cout << “input x: ” << endl;
cin >> x;
cout << “input y: ” << endl;
cin >> y;
cout << x << ” + ” << y << ” = ” << x + y << endl;
return 0;
}
supprocess模塊提供了一些函數,方便我們用於創建進程。
subprocess.call(*popenargs, **kwargs)
執行命令。該函數將一直等待到子進程執行結束,並返回進程的returncode。文章一開始的樣例就演示了call函數。假設子進程不須要進行交 互,就能夠使用該函數來創建。
subprocess.check_call(*popenargs, **kwargs)
與subprocess.call(*popenargs, **kwargs)功能一樣,僅僅是假設子進程返回的returncode不為0的話,將觸發CalledProcessError異常。在異常對象中,包 括進程的returncode信息。
subprocess模塊的內容就這么多。在Python手冊中,還介紹了怎樣使用 subprocess來替換一些老的模塊,老的函數的樣例。趕興趣的朋友能夠看一下。
參考文檔:
subprocess — Subprocess management
PyMoTW:subprocess
http://hi.baidu.com/kobeantoni/blog/item/a034bce9d0e01bdfd539c9a4.html