subprocess模块使用

从 os.system(cmd) 和 os.popen(cmd) 的官方文档都引导我们去使用 subprocess 模块

关于 subprocess 模块的用法基本就是讲述 subprocess.run()

subprocess.run(args, *, stdin=None, input=None, stdout=None, stderr=None, capture_output=False, shell=False, cwd=None, timeout=None, check=False, encoding=None, errors=None, text=None, env=None, universal_newlines=None)

subprocess.run() 是在 Python 3.5 加入的,之前的版本要调用 subprocess.call() 方法。

基本用法

import subprocess

p1 = subprocess.run('ls')  # 这行与终端执行 ls 效果一样,直接输出 ls 的结果
print('****\n', p1, p1.stdout)
scratch.py  scratch_1.py  ****  CompletedProcess(args='ls', returncode=0) None

我们看到 p1 的类型是一个 CompletedProcess , 此时 p1.stdout 的内容是 None ,也就是说我们没有捕获到命令的执行输出

捕获命令输出


p1 = subprocess.run('ls', capture_output=True)
print('****\n', type(p1.stdout), p1.stdout)  # p1.stdout.decode() 显示文本
****  <class 'bytes'> b'scratch.py\nscratch_1.py\n'

加了 capture_output=True 参数,命令的执行结果不直接打印到控制台,而是可以通过 p1.stdout 获得的

capture_output=True 实质上会设置两个参数值 stdout=subprocess.PIPE, stderr=subprocess.PIPE ,也就是它同时也会捕获标准错误输出,放置到 p1.stderr 中。如果只需捕获标准输出的话,只要 stdout=subprocess.PIPE

默认 p1.stdout 是一个 bytes 类型,要显示为文本需要用 p1.stdout.decode() ,或者再加上参数 text=True 无需 stdout.decode() 了。用 encoding="utf-8" 也能免除 decode() 。

p1 = subprocess.run('ls', capture_output=True, text=True)
print('****\n', type(p1.stdout), '\n', p1.stdout)
****  <class 'str'>   scratch.py  scratch_1.py

加了 text=True 后, p1.stdout 的类型也变成了字符串 str 。

命令参数与 shell 命令

前面的命令 ls 没有带参数,那怎么执行带参数的命令呢?先尝试着把命令与参数写在一块

subprocess.run('ls -la')
FileNotFoundError: [Errno 2] No such file or directory: 'ls -la': 'ls -la'

subprocess.run() 并不知道 ls -la 中的 ls 是命令,其余为参数,而是把整个字符串包括中间的空间当作命令,所以提示找不到 ls -la 这样的命令。解决方法有一,加上 shell=True 参数

subprocess.run('ls -la', shell=True)
total 16  drwxr-xr-x 4 yanbin 2022363315 128 Aug 12 21:17 .  drwxr-xr-x 11 yanbin 2022363315 352 Aug 12 20:49 ..  -rw-r--r-- 1 yanbin 2022363315 933 Aug 12 09:59 scratch.py  -rw-r--r-- 1 yanbin 2022363315 56 Aug 12 21:17 scratch_1.py

有了 shell=True 的话, run() 的第一个参数可以写的更复杂,管道操作也行,如下
subprocess.run('ls -la; pwd && ls | grep scratch', shell=True)

但 shell=True 不那么安全,比如说对于 ls 命令,它后面的目录部分是用户输入的话就可被注入

subprocess.run(f'ls -la ${folder}', shell=True)
假如用户输入的 folder="; rm -rf / , 就能随意的干活了。当 shell=True 是可以借助于 shlex.quote() 函数来抵御不安全的输入。

为安全起见,我们可保持默认的 shell=False 值,命令参数则必须以列表形式提供
subprocess.run(['ls', '-la', folder])
列表第一个元素为命令,其他为参数,这时候 folder 能够防御住攻击了。

输出重定向(文件或设备)

with open('output.txt', 'w') as file:
    subprocess.run(['ls', '-la'], stdout=file, text=True)

现在命令的输出进到文件 output.txt 中了。要让标准错误输出也重定向到文件的话,就加上 stderr=file ,或者 stdout=file, stderr=subprocess.STDOUT 。这一连贯的操作就好比执行如下的 shell 命令

with open('output.txt', 'w') as file:
    p1 = subprocess.run(['ls', '-la', 'lala'], stdout=file, stderr=subprocess.STDOUT,text=True)
    print(p1.returncode)

上面的代码效果与这个 shell 命令差不多, 2 为标准错误输出,1 为标准输出
ls -la lala> output.txt 2>&1
由于 lala 目录不存在,所以 p1.returncode 为 1 , 非零即为有错误,看看 output.txt 中的内容为

ls: abc: No such file or directory
如果想忽略错误输出,例如一个程序中既有标准输出又有标准错误输出,且返回状态码为 2,用下面 Python 程序来模拟

#!/usr/bin/env python3
 
import sys
 
print('hello stdout')
print('hello stderr', file=sys.stderr)
exit(2)

想要执行出下方第四行中命令的效果

$ ./test.py
hello stdout
hello stderr
$ ./test.py > output.txt 2> /dev/null
$ echo $?
2
$ cat output.txt
hello stdout

相应的 subprocess.run() 代码如下

with open('output.txt', 'w') as file:
    p1 = subprocess.run('./test.py', stdout=file, text=True, stderr=subprocess.DEVNULL)
    print(p1.returncode) # 这时候的 p1.stderr 就是 None 报告错误

如果我们捕获了输出后, subprocess.run() 执行命令出错时不会报告给 Python 解释器

p1 = subprocess.run(['ls', '-la', 'lala'], capture_output=True, text=True)

lala 目录不存在,但上面的代码会没有任何输出,我们只得在执行后检查 p1 的返回码及 p1.stderr 的值

if p1.returncode != 0:
    print(p1.stderr)

输出

ls: lala: No such file or directory

要是加上 check=True 参数就不一样的了

subprocess.run(['ls', '-la', 'lala'], capture_output=True, text=True, check=True)

直接报告异常

Traceback (most recent call last):  File "/Users/yanbin/Library/Preferences/PyCharmCE2019.1/scratches/scratch_1.py", line 4, in <module>  p1 = subprocess.run(['ls', '-la', 'lala'], capture_output=True, text=True, check=True)  File "/usr/local/Cellar/python/3.7.3/Frameworks/Python.framework/Versions/3.7/lib/python3.7/subprocess.py", line 487, in run  output=stdout, stderr=stderr)  subprocess.CalledProcessError: Command '['ls', '-la', 'lala']' returned non-zero exit status 1.

这对于希望在有任何命令执行异常时退出很有用

管道操作

前面提到过,用 subprocess.run() 设置 shell=True 时,命令中可以随意使用复杂的 shell 命令,包括管道操作。如果 shell=False 默认值时,管道操作就需要显式的执行多个 subprocess.run() 操作并且通过像 input=p1.stdout 这样串联起来,这原本就是管道的意义

p1 = subprocess.run(['cat', 'a.json'], capture_output=True, text=True)
p2 = subprocess.run(['grep', 'id'], capture_output=True, text=True, input=p1.stdout)
print(p2.stdout)

相当于执行了 shell 命令

cat a.json | grep id

其他

encoding 和 errors 参数是 Python 3.6 加入的。 encoding='utf-8' 与 text=True 有着类似的功效

text, universal_newlines 和 capture_output 是 Python 3.7 加入的,之前版本捕获输出就要用 stdout=subprocess.PIPE , 且得到的 stdout 是 bytes 类型,stderr 相类似。

timeout=<in seconds> 会在多少秒之后还没执行完命令后得到 TimeoutExpired 异常,并且终止掉该命令的执行

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,233评论 6 495
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,357评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,831评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,313评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,417评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,470评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,482评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,265评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,708评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,997评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,176评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,827评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,503评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,150评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,391评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,034评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,063评论 2 352