subprocess.Popen用来创建子进程。
当我们需要调用系统的命令的时候,最先考虑的os模块。用os.system()和os.popen()来进行操作。但是这两个命令过于简单,不能完成一些复杂的操作,如给运行的命令提供输入或者读取命令的输出,判断该命令的运行状态,管理多个命令的并行等等。这时subprocess中的Popen命令就能有效的完成我们需要的操作。
1)Popen启动新的进程与父进程并行执行,默认父进程不等待新进程结束。
defTestPopen():importsubprocess
p=subprocess.Popen("dir",shell=True)foriinrange(250) :print("other things")
2)p.wait函数使得父进程等待新创建的进程运行结束,然后再继续父进程的其他任务。且此时可以在p.returncode中得到新进程的返回值。
defTestWait():importsubprocessimportdatetimeprint(datetime.datetime.now())
p=subprocess.Popen("sleep 10",shell=True)
p.wait()print(p.returncode)print(datetime.datetime.now())
3) p.poll函数可以用来检测新创建的进程是否结束。
defTestPoll():importsubprocessimportdatetimeimporttimeprint(datetime.datetime.now())
p=subprocess.Popen("sleep 10",shell=True)
t=1while(t<=5):
time.sleep(1)
p.poll()print(p.returncode)
t+=1print(datetime.datetime.now())
4) p.kill或p.terminate用来结束创建的新进程,在windows系统上相当于调用TerminateProcess(),在posix系统上相当于发送信号SIGTERM和SIGKILL。
defTestKillAndTerminate():
p=subprocess.Popen("notepad.exe")
t=1while(t<=5):
time.sleep(1)
t+=1p.kill()#p.terminate()print("new process was killed")
5) p.communicate可以与新进程交互,但是必须要在popen构造时候将管道重定向。
defTestCommunicate():importsubprocess
cmd="dir"p=subprocess.Popen(cmd, shell=True, stdout=subprocess.PIPE, stderr=subprocess.STDOUT)
(stdoutdata, stderrdata)=p.communicate()ifp.returncode!=0:print(cmd+"error !")#defaultly the return stdoutdata is bytes, need convert to str and utf8forrinstr(stdoutdata,encoding='utf8').split("\n"):print(r)print(p.returncode)defTestCommunicate2():importsubprocess
cmd="dir"#universal_newlines=True, it means by text way to open stdout and stderrp=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)
6) call函数可以认为是对popen和wait的分装,直接对call函数传入要执行的命令行,将命令行的退出code返回。
defTestCall():
retcode=subprocess.call("c:\\test.bat")print(retcode)
7)subprocess.getoutput 和 subprocess.getstatusoutput
,基本上等价于subprocess.call函数,但是可以返回output,或者同时返回退出code和output。
但是可惜的是好像不能在windows平台使用,在windows上有如下错误:'{'isnotrecognizedasaninternalorexternal
command,operable
programorbatch
file.
defTestGetOutput():
outp=subprocess.getoutput("ls -la")print(outp)defTestGetStatusOutput():
(status, outp)=subprocess.getstatusoutput('ls -la')print(status)print(outp)
8)总结
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.*