Subprocess模块(* * * *)
当我们需要调用系统的命令的时候,最先考虑的os模块。用os.system()和os.popen()来进行操作。但是这两个命令过于简单,不能完成一些复杂的操作,如给运行的命令提供输入或者读取命令的输出,判断该命令的运行状态,管理多个命令的并行等等。这时subprocess中的Popen命令就能有效的完成我们需要的操作。
subprocess模块允许一个进程创建一个新的子进程,通过管道连接到子进程的stdin/stdout/stderr,获取子进程的返回值等操作。
The subprocess module allows you to spawn new processes, connect to their input/output/error pipes, and obtain their return codes.
This module intends to replace several other, older modules and functions, such as: os.system、os.spawn*、os.popen*、popen2.*、commands.*
这个模块一个类:Popen。
1
2
3
|
#Popen它的构造函数如下: subprocess.Popen(args, bufsize = 0 , executable = None , stdin = None , stdout = None ,stderr = None , preexec_fn = None , close_fds = False , shell = False ,<br> cwd = None , env = None , universal_newlines = False , startupinfo = None , creationflags = 0 ) |
# 参数args可以是字符串或者序列类型(如:list,元组),用于指定进程的可执行文件及其参数。 # 如果是序列类型,第一个元素通常是可执行文件的路径。我们也可以显式的使用executeable参 # 数来指定可执行文件的路径。在windows操作系统上,Popen通过调用CreateProcess()来创 # 建子进程,CreateProcess接收一个字符串参数,如果args是序列类型,系统将会通过 # list2cmdline()函数将序列类型转换为字符串。 # # # 参数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()函数,用于设置子进程的一些属性,如:主窗口的外观,进程的优先级等等。
简单命令:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
|
import subprocess a = subprocess.Popen( ‘ls‘ ) # 创建一个新的进程,与主进程不同步 print ( ‘>>>>>>>‘ ,a) #a是Popen的一个实例对象 ‘‘‘ >>>>>>> <subprocess.Popen object at 0x10185f860> __init__.py __pycache__ log.py main.py ‘‘‘ # subprocess.Popen(‘ls -l‘,shell=True) # subprocess.Popen([‘ls‘,‘-l‘]) |
subprocess.PIPE
在创建Popen对象时,subprocess.PIPE可以初始化stdin, stdout或stderr参数。表示与子进程通信的标准流。
1
2
3
4
5
6
|
import subprocess # subprocess.Popen(‘ls‘) p = subprocess.Popen( ‘ls‘ ,stdout = subprocess.PIPE) #结果跑哪去啦? print (p.stdout.read()) #这这呢:b‘__pycache__\nhello.py\nok.py\nweb\n‘ |
这是因为subprocess创建了子进程,结果本在子进程中,if 想要执行结果转到主进程中,就得需要一个管道,即 : stdout=subprocess.PIPE
subprocess.STDOUT
创建Popen对象时,用于初始化stderr参数,表示将错误通过标准输出流输出。
Popen的方法
View Code
supprocess模块的工具函数
1
2
3
4
5
6
7
8
9
10
11
12
13
|
supprocess模块提供了一些函数,方便我们用于创建进程来实现一些简单的功能。 subprocess.call( * popenargs, * * kwargs) 运行命令。该函数将一直等待到子进程运行结束,并返回进程的returncode。如果子进程不需要进行交 互,就可以使用该函数来创建。 subprocess.check_call( * popenargs, * * kwargs) 与subprocess.call( * popenargs, * * kwargs)功能一样,只是如果子进程返回的returncode不为 0 的话,将触发CalledProcessError异常。在异常对象中,包 括进程的returncode信息。 check_output( * popenargs, * * kwargs) 与call()方法类似,以byte string的方式返回子进程的输出,如果子进程的返回值不是 0 ,它抛出CalledProcessError异常,这个异常中的returncode包含返回码,output属性包含已有的输出。 getstatusoutput(cmd) / getoutput(cmd) 这两个函数仅仅在Unix下可用,它们在shell中执行指定的命令cmd,前者返回(status, output),后者返回output。其中,这里的output包括子进程的stdout和stderr。 |
import subprocess #1 # subprocess.call(‘ls‘,shell=True) ‘‘‘ hello.py ok.py web ‘‘‘ # data=subprocess.call(‘ls‘,shell=True) # print(data) ‘‘‘ hello.py ok.py web 0 ‘‘‘ #2 # subprocess.check_call(‘ls‘,shell=True) ‘‘‘ hello.py ok.py web ‘‘‘ # data=subprocess.check_call(‘ls‘,shell=True) # print(data) ‘‘‘ hello.py ok.py web 0 ‘‘‘ # 两个函数区别:只是如果子进程返回的returncode不为0的话,将触发CalledProcessError异常 #3 # subprocess.check_output(‘ls‘)#无结果 # data=subprocess.check_output(‘ls‘) # print(data) #b‘hello.py\nok.py\nweb\n‘
交互命令:
终端输入的命令分为两种:
- 输入即可得到输出,如:ifconfig
- 输入进行某环境,依赖再输入,如:python
需要交互的命令示例
待续