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