python --subprocess

從Python 2.4開始,Python引入subprocess模塊來管理子進程,以取代一些舊模塊的方法:如 os.system、os.spawn*、os.popen*、popen2.*、commands.*不但能夠調用外部的命令做爲子進程,並且能夠鏈接到子進程的input/output/error管道,獲取相關的返回信息html

運行python的時候,咱們都是在建立並運行一個進程。像Linux進程那樣,一個進程能夠fork一個子進程,並讓這個子進程exec另一個程序。在Python中,咱們經過標準庫中的subprocess包來fork一個子進程,並運行一個外部的程序。
subprocess包中定義有數個建立子進程的函數,這些函數分別以不一樣的方式建立子進程,因此咱們能夠根據須要來從中選取一個使用。另外subprocess還提供了一些管理標準流(standard stream)和管道(pipe)的工具,從而在進程間使用文本通訊。python

subprocess以及經常使用的封裝函數shell

1)Popen啓動新的進程與父進程並行執行,默認父進程不等待新進程結束。windows


import subprocesscurl

def TestPopen():
  p=subprocess.Popen("dir",shell=True)
  for i in range(10) :
    print "other things: %d"%i

TestPopen()ide

 

2)p.wait函數使得父進程等待新建立的進程運行結束,而後再繼續父進程的其餘任務。且此時能夠在p.returncode中獲得新進程的返回值。函數

import subprocess
child = subprocess.Popen(["ping","www.baidu.com"])
child.wait()
print("parent process")工具

 

3) p.poll函數能夠用來檢測新建立的進程是否結束。url


import subprocess
import datetime
import timespa

def TestPoll():
print (datetime.datetime.now())
p=subprocess.Popen(["ping","www.baidu.com"],shell=True)
t = 1
while(t <= 3):
if(p.poll()==None):
print (p.returncode)
break
t+=1
print (datetime.datetime.now())

TestPoll()

 

4) p.kill或p.terminate用來結束建立的新進程,在windows系統上至關於調用TerminateProcess(),在posix系統上至關於發送信號SIGTERM和SIGKILL。

import subprocess
import time

def TestKillAndTerminate():
    p=subprocess.Popen("notepad.exe")
    t = 1
    while(t <= 5):
      time.sleep(1)
      t +=1
    p.kill()
    #p.terminate()
    print ("new process was killed")

TestKillAndTerminate()

 

5) p.communicate能夠與新進程交互,可是必需要在popen構造時候將管道重定向。(要注意的是,communicate()是Popen對象的一個方法,該方法會阻塞父進程,直到子進程完成。)

def TestCommunicate():
    cmd = "dir"
    p=subprocess.Popen(cmd, shell=True, stdout=subprocess.PIPE, stderr=subprocess.STDOUT)
    (stdoutdata, stderrdata) = p.communicate()
    print (stdoutdata)
    print ">>>>>>>>>>>>>>>>>>>>>>"
    if p.returncode != 0:
        print (cmd + "error !")
    #defaultly the return stdoutdata is bytes, need convert to str 
    for r in str(stdoutdata).split("\n"):
        print (r)
        
    print (p.returncode)


def TestCommunicate2():
    cmd = "dir"
    #universal_newlines=True, it means by text way to open stdout and stderr
    p = subprocess.Popen(cmd, shell=True, universal_newlines=True, stdout=subprocess.PIPE, stderr=subprocess.STDOUT)
    curline = p.stdout.readline()
    while(curline != ""):
        print (curline)
        curline = p.stdout.readline()
    p.wait()
    print (p.returncode)


TestCommunicate()
TestCommunicate2()

 

6) call函數能夠認爲是對popen和wait的封裝,直接對call函數傳入要執行的命令行,將命令行的退出code返回。

import subprocess

def TestCall():
    retcode = subprocess.call("dir",shell=True)
    print (retcode)
    
TestCall()

 

 

7)總結

popen的參數,第一個爲字符串(或者也能夠爲多個非命名的參數),表示你要執行的命令和命令的參數;後面的均爲命名參數;shell=True,表示你前面的傳入的命令將在shell下執行,若是你的命令是個可執行文件或bat,不須要指定此參數;stdout=subprocess.PIPE用來將新進程的輸出重定向,stderr=subprocess.STDOUT將新進程的錯誤輸出重定向到stdout,stdin=subprocess.PIPE用來將新進程的輸入重定向;universal_newlines=True表示以text的方式打開stdout和stderr。 

 

其餘的不推薦使用的模塊:

os.system
os.spawn*
os.popen*
popen2.*
commands.*

 

參考:

http://www.liuzhongshu.com/code/subprocess-detail.html

http://www.liuzhongshu.com/code/python-vs-bat.html

 

總結

subprocess.call, subprocess.check_call(), subprocess.check_output()

subprocess.Popen(), subprocess.PIPE

Popen.wait(), Popen.communicate()

 

 

總結

subprocess.call, subprocess.check_call(), subprocess.check_output()

subprocess.Popen(), subprocess.PIPE

Popen.wait(), Popen.communicate()

相關文章
相關標籤/搜索