1. 程式人生 > >Python_cmd的各種實現方法及優劣(subprocess.Popen, os.system和commands.getstatusoutput)

Python_cmd的各種實現方法及優劣(subprocess.Popen, os.system和commands.getstatusoutput)

轉自:http://blog.csdn.net/menglei8625/article/details/7494094

目前我使用到的python中執行cmd的方式有三種:

1. 使用os.system("cmd")

這是最簡單的一種方法,特點是執行的時候程式會打出cmd在linux上執行的資訊。使用前需要import os。

[python] 
  1. os.system("ls")  

2. 使用Popen模組產生新的process

現在大部分人都喜歡使用Popen。Popen方法不會打印出cmd在linux上執行的資訊。的確,Popen非常強大,支援多種引數和模式。使用前需要from subprocess import Popen, PIPE。但是Popen函式有一個缺陷,就是它是一個阻塞的方法。如果執行cmd時產生的內容非常多,函式非常容易阻塞住。解決辦法是不使用wait()方法,但是也不能獲得執行的返回值了。

Popen原型是:

[python] 
  1. 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)  

引數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。

例如:

[python]
  1. p = Popen("cp -rf a/* b/", shell=True, stdout=PIPE, stderr=PIPE)  
  2. p.wait()  
  3. if p.returncode != 0:  
  4.     print"Error."
  5.     return -1

3. 使用commands.getstatusoutput方法

這個方法也不會打印出cmd在linux上執行的資訊。這個方法唯一的優點是,它不是一個阻塞的方法。即沒有Popen函式阻塞的問題。使用前需要import commands。

例如:

[python]
  1. status, output = commands.getstatusoutput("ls")  

還有隻獲得output和status的方法:
[python]
  1. commands.getoutput("ls")  
  2. commands.getstatus("ls")