文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

【module】subprocess

2023-01-31 08:20

关注

subprocess模块是python从2.4版本开始引入的模块。主要用来取代 一些旧的模块方法,如os.system、os.spawn、os.popen、commands.*等subprocess通过子进程来执行外部指令,并通过input/output/error管道,获取子进程的执行的返回信息。

使用方法:

运行外部命令:subprocess.call(command)

subprocess的call方法可以用于执行一个外部命令,但该方法不能返回执行的结果,只能返回执行的状态码: 成功(0) 或 错误(非0)

方法一:

subprocess.call(['ls','-l'])
总用量 4
-rw-r--r-- 1 liangml liangml  0 3月   7 15:38 mark.md
-rw-r--r-- 1 liangml liangml 68 3月  11 13:39 php-fpm.py
0

方法二:

subprocess.call("ls -l",shell=True)
总用量 4
-rw-r--r-- 1 liangml liangml  0 3月   7 15:38 mark.md
-rw-r--r-- 1 liangml liangml 68 3月  11 13:39 php-fpm.py
0

上面示例是在终端中运行,虽然可以看到运行结果,但实际取值只是状态码

a = subprocess.call("ls -l",shell=True)
总用量 4
-rw-r--r-- 1 liangml liangml  0 3月   7 15:38 mark.md
-rw-r--r-- 1 liangml liangml 81 3月  11 13:44 php-fpm.py
print(a)
0

错误处理:subprocess.check_call()

我们说过call执行返回一个状态码,我们可以通过check_call()函数来检测命令的执行结果,如果不成功将返回 subprocess.CalledProcessError 异常

示例:

import subprocess
try:
    subprocess.check_call("sdf",shell=True)
except subprocess.CalledProcessError as err:
    print("Command Error",err)

捕获输出结果:subprocess.check_output()

call()方法启动的进程,其标准输入输出会绑定到父进程的输入和输出。调用程序无法获取命令的输出结果。但可以通过check_output()方法来捕获输出。

示例:

output = subprocess.check_output("ls -l",shell=True)
print(output.decode('utf-8'))
import subprocess
try:
    output = subprocess.check_output("lT -l",shell=True,stderr=subprocess.STDOUT)
except subprocess.CalledProcessError as err:
    print("Command Error",err)
#执行结果
Command Error Command 'lT -l' returned non-zero exit status 127

直接处理管道:subprocess.Popen()

函数call(), check_call() 和 check_output() 都是Popen类的包装器。直接使用Popen会对如何运行命令以及如何处理其输入输出有更多控制。如通过为stdin, stdout和stderr传递不同的参数。

  • 与进程的单向通信:通过Popen()方法调用命令后执行的结果,可以设置stdout值为PIPE,再调用communicate()获取结果,返回结果为tuple. 在python3中结果为byte类型,要得到str类型需要decode转换一下

输出结果(读)

# 直接执行命令输出到屏幕
>>> subprocess.Popen("ls -l",shell=True)
<subprocess.Popen object at 0x7fe38cd67f98>
总用量 4
-rw-r--r-- 1 liangml liangml  0 3月   7 15:38 mark.md
-rw-r--r-- 1 liangml liangml 76 3月  11 13:59 php-fpm.py
# 不输出到屏幕,输出到变量
>>> proc = subprocess.Popen(['echo','"Stdout"'],stdout=subprocess.PIPE)
# communicate返回标准输出或标准出错信息
>>> stdout_value = proc.communicate()
>>> stdout_value
(b'"Stdout"\n', None)



>>> proc = subprocess.Popen(['ls','-l'],stdout=subprocess.PIPE)
>>> stdout_value = proc.communicate()
>>> stdout_value
(b'\xe6\x80\xbb\xe7\x94\xa8\xe9\x87\x8f 48\ndrwxr-xr-x  3 liangml liangml 4096 3\xe6\x9c\x88  10 14:57 Desktop\ndrwxr-xr-x  6 liangml liangml 4096 3\xe6\x9c\x88   4 22:00 Documents\ndrwxr-xr-x  2 liangml liangml 4096 3\xe6\x9c\x88   8 18:32 Downloads\ndrwxr-xr-x  3 liangml liangml 4096 3\xe6\x9c\x88   3 11:10 GitBook\ndrwxr-xr-x  3 liangml liangml 4096 2\xe6\x9c\x88  22 17:52 Music\ndrwxr-xr-x  3 liangml liangml 4096 2\xe6\x9c\x88  22 17:48 Pictures\ndrwxr-xr-x  3 liangml liangml 4096 2\xe6\x9c\x88  26 18:29 PycharmProjects\ndrwxr-xr-x  3 liangml liangml 4096 2\xe6\x9c\x88  22 18:01 Steam\ndrwxr-xr-x 10 liangml liangml 4096 2\xe6\x9c\x88  27 11:21 svn\ndrwxr-xr-x  2 liangml liangml 4096 2\xe6\x9c\x88  22 17:48 Templates\ndrwxr-xr-x  2 liangml liangml 4096 2\xe6\x9c\x88  22 17:24 Videos\ndrwxr-xr-x  3 liangml liangml 4096 3\xe6\x9c\x88   7 21:00 vnote_notebooks\n', None)




# 结果输出到文件
>>> file_handle = open('/home/liangml/t.log','w+')
>>> subprocess.Popen('ls -l',shell=True,stdout=file_handle)
>>> subprocess.call("ls -l",shell=True)
总用量 52
-rw-r--r--  1 liangml liangml  779 3月  11 14:14 t.log
vnote_notebooks

与进程的双向通信

proc = subprocess.Popen('cat', shell=True, stdin=subprocess.PIPE, stdout=subprocess.PIPE)
msg = 'hello world'.encode('utf-8')
# 写入到输入管道
proc.stdin.write(msg)
11  # 输入结果
stdout_value = proc.communicate()
stdout_value
(b'hello world', None)  # 输出结果

# 在需要进行相互交互的输入输出过程也可以使用shtin来实现
# 以下实现打开python3的终端,执行一个print命令
proc = subprocess.Popen(['python3'],stdin=subprocess.PIPE,stdout=subprocess.PIPE, stderr=subprocess.PIPE,)
proc.stdin.write('print("helloworld")'.encode('utf-8'))
out_value,err_value=proc.communicate()
print(out_value)
b'helloworld\n'
print(err_value)
b''

Popen.communicate()方法用于和子进程交互:发送数据到stdin,并从stdout和stderr读数据,直到收到EOF。等待子进程结束。

捕获错误输出

proc = subprocess.Popen(['python3'],stdin=subprocess.PIPE,stdout=subprocess.PIPE, stderr=subprocess.PIPE,)
proc.stdin.write('print "helloworld"'.encode('utf-8'))
18
out_value,err_value=proc.communicate()
out_value
b''
print(err_value.decode('utf-8'))
  File "<stdin>", line 1
    print "helloworld"
                     ^
SyntaxError: Missing parentheses in call to 'print'

Popen其他方法

  • Popen.pid 查看子进程ID
  • Popen.returncode 获取子进程状态码,0表示子进程结束,None未结束

在使用Popen调用系统命令式,建议使用communicate与stdin进行交互并获取输出(stdout),这样能保证子进程正常退出而避免出现僵尸进程:

示例:

proc = subprocess.Popen('ls -l', shell=True, stdout=subprocess.PIPE)
# 当前子进程ID
proc.pid
5919
# 返回状态None,进程未结束
print(proc.returncode)
None
# 通过communicate提交后
out_value = proc.communicate()
proc.pid
5919
# 返回状态为0,子进程自动结束
print(proc.returncode)
0
阅读原文内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     813人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     354人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     318人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     435人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     224人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯