subprocess 模块
功能:用来生成子进程,并可以通过管道连接它们的输入/输出/错误,以及获得它们的返回值。
它用来代替多个旧模块和函数:
os.system
os.spawn*
os.popen*
popen2.*
commands.*
目的:启动一个新的进程并且与之通信,subprocess包中定义有数个创建子进程的函数,这些函数分别以不同的方式创建子进程。
注意:在python3中此模块返回的是bytes类型,在python2中返回的是字符串。
以下为一些常用方法:
check_output():执行命令将结果返回给变量,不做屏幕打印
getoutput():执行命令返回结果
call():父进程等待子进程完成,返回命令执行状态(0成功,非0失败)
getstatusoutput():执行命令以元组的形式返回,元组内容包含(第一个元素为执行状态,第二个命令为执行的结果)
run():直接运行命令。说明:此方法在p3.5前没有
说明:以上方法中最常用的是:getstatusoutput()
如果出现需要命令进行交互的操作,要使用Popen方法
注意:以上标识的方法在底层都是封装的Popen,以下说明及使用
语法:
class 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:shell命令,可以是字符串,或者序列类型(列表,元组)
bufsize:指定缓冲。0无缓冲,1缓冲1行,其他 缓冲区大小,负值系统缓冲
stdin,stdout,stderr:标准输入,标准输出,标准错误
preexec_fn:只在unix下有效,用于指定一个可执行对象,它将在子进程运行之前被调用
close_sfs:在windows平台下设置为True,则新创建的子进程不会继承父进程的输入,输出,错误管道。
shell:为True时,指定将args以shell命令运行
cwd:设置子进程的当前目录。obj = subprocess.Popen(['pwd'],stdin=subprocess.PIPE,stdout=subprocess.PIPE,cwd='/tmp')
env:指定子进程的环境变量,如env=None,子进程的环境变量将从父进程中继承。
universal_newlines:不同系统的换行符不同,默认已经兼容
startupinfo与createProcess()函数:设置子进程的一些属性,如主窗口的外观,进程的优先级等等(可以忘记)
根据popen方法进行关联使用的方法如下:
poll(): 检查子进程运行状态,res.poll() 返回None标识命令未执行完,返回0标识执行完成。
举例:res = subprocess.Popen('sleep 10;echo "hello"',shell=True,stdout=subprocess.PIPE,stderr=subprocess.PIPE)
res.poll()
wait(): Popen对象创建后,主程序不会自动等待子进程完成。我们必须调用对象的wait()方法,父进程才会等待 (也就是阻塞block)。
kill():终止子进程
pid:获取子进程的进程ID
child.send_signal() : 向子进程发送信号(用处少)
child.terminate() : 终止子进程(用处少)
communicate():在命令执行过程中,再给它传数据
stdin:标准输入
stdout:标准输出
stderr:标准错误
Popen()建立子进程的时候改变标准输入、标准输出和标准错误,并可以利用subprocess.PIPE将多个子进程的输入和输出连接在一起,构成管道(pipe)
举例1:单独使用stdout
import subprocess
out = subprocess.Popen('ls -l',shell=True,stdout=subprocess.PIPE)
out.stdout.read().decode()
注意:这里我单独添加了decode解码,因为p3通过read出来的都是bytes类型字符,需要通过解码才能显示看懂。
举例2:通过stdout输出,作为stdin的输入
import subprocess
out = subprocess.Popen('cat /etc/passwd',shell=True,stdout=subprocess.PIPE)
mess = subprocess.Popen('grep 0:0',shell=True,stdin=out.stdout,stdout=subprocess.PIPE)
原理:subprocess.PIPE实际上为文本流提供一个缓存区。out的stdout将文本输出到缓存区,随后mess的stdin从该PIPE中将文本读取走。
mess的输出文本也被存放在PIPE中
举例:通过python进行shell 的sudo自动输入密码
subprocess.Popen('echo "abc123" | sudo -S ls -l',shell=True,stdout=subprocess.PIPE)
总结:
直接执行shell命令,无交互的使用:getstatusoutput() 方法,返回元组(状态码,结果)
执行shell命令出现交互的举例:
#test1.py
import sys
line = sys.stdin.readline()
print 'test',line
#run.py
from subprocess import *
p =Popen('./test1.py',stdin=PIPE,stdout=PIPE)
p.stdin.write('say hi/n')
print p.stdout.readline()
#result
test say hi