可以执行shell命令的相关模块和函数有:

  • os.system
  • os.spawn*
  • os.popen*          --废弃
  • popen2.*           --废弃
  • commands.*      --废弃,3.x中被移除

上面这些命令,可以使用subprocess完美的实现,而且具有丰富的功能:

call:   python3.5以下才有, python3.5及以上变成run方法

执行命令,返回状态码


>>> a = subprocess.call('whoami')
huangxm-pc\huangxm
>>> print(a)
0

执行一个带参数的命令

>>> subprocess.call('ls -l')
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>

报错了,对于这种可以加上shell=True, 表示完全当成shell命令执行

>>> subprocess.call('ls -l', shell=True)
total 48
drwxr-xr-x 5 huanghao huanghao 4096 Mar 12 18:42 day7
drwxrwxrwx 2 huanghao huanghao 4096 Oct 19 22:42 Desktop

check_call

执行命令,如果执行状态码是 0 ,则返回0,否则抛异

>>> subprocess.check_call('cat /etc/passwd | grep root', shell=True)
root:x:0:0:root:/root:/bin/bash
0
>>> subprocess.check_call('cat /etc/passwddd | grep root', shell=True)  #执行失败,抛出异常
cat: /etc/passwddd: No such file or directory
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
  File "/usr/lib/python3.4/subprocess.py", line 557, in check_call
    raise CalledProcessError(retcode, cmd)
subprocess.CalledProcessError: Command 'cat /etc/passwddd | grep root' returned non-zero exit status 1

check_output

执行命令,如果执行状态码是 0 ,则返回0,否则抛异

 

subprocess.Popen(...)

用于执行复杂的系统命令

参数:

  • args:shell命令,可以是字符串或者序列类型(如:list,元组)
  • bufsize:指定缓冲。0 无缓冲,1 行缓冲,其他 缓冲区大小,负值 系统缓冲
  • stdin, stdout, stderr:分别表示程序的标准输入、输出、错误句柄
  • preexec_fn:只在Unix平台下有效,用于指定一个可执行对象(callable object),它将在子进程运行之前被调用
  • close_sfs:在windows平台下,如果close_fds被设置为True,则新创建的子进程将不会继承父进程的输入、输出、错误管道。
    所以不能将close_fds设置为True同时重定向子进程的标准输入、输出与错误(stdin, stdout, stderr)。
  • shell:同上
  • cwd:用于设置子进程的当前目录
  • env:用于指定子进程的环境变量。如果env = None,子进程的环境变量将从父进程中继承。
  • universal_newlines:不同系统的换行符不同,True -> 同意使用 \n
  • startupinfo与createionflags只在windows下有效
    将被传递给底层的CreateProcess()函数,用于设置子进程的一些属性,如:主窗口的外观,进程的优先级等等
a = subprocess.Popen('ipconfig', shell=True)
print(a)

执行结果
<subprocess.Popen object at 0x0000000002809240><subprocess.Popen object at 0x0000000002809240>

命令成功执行了,但是没有获得到结果,怎么获得结果呢?

a = subprocess.Popen('ipconfig', shell=True, stdout=subprocess.PIPE).stdout.read()
print(str(a, 'gbk'))      #winodws中文是gbk

有时候我们并不是只这样运行命令,而是需要进入一些目录去执行一些命令,比如我们要在D盘建立一个文件夹:

a = subprocess.Popen('mkdir aaaaa', shell=True, cwd='C:/')

执行之后,到C盘看一下,有没有一个aaaaa的文件夹。

 

再来看一个复杂点的:

第一步:先在D盘根目录建立一个文件aa.py,内容如下:

def add():
    x = input('input x:').strip()
    print('\n')
    y = input('input y:').strip()
    print('\n')
    print('the result is ', int(x)+int(y))

if __name__ == "__main__":
    add()

第二步:执行下面的代码

obj = subprocess.Popen('python3 D:/aa.py', shell=True, cwd='c:/python34', stdin=subprocess.PIPE, stdout=subprocess.PIPE, stderr=subprocess.PIPE)
obj.stdin.write(b'1\n')
obj.stdin.write(b'2\n')
obj.stdin.close()
cmd_out = str(obj.stdout.read(), 'gbk')
obj.stdout.close()
print(cmd_out)

第三步:查看执行结果

input x:

input y:

the result is  3

在提示要输入的时候,我们根本就没有在键盘上输入,因为stdin.write已经帮我们完成了输入。 然后程序通过stdout.read()将结果读取出来。如果程序执行中发生错误,还可以能过obj.stderr.read()获取错误信息。