Python subprocess模块学习总结


Posted in Python onMarch 13, 2014

一、subprocess以及常用的封装函数
运行python的时候,我们都是在创建并运行一个进程。像Linux进程那样,一个进程可以fork一个子进程,并让这个子进程exec另外一个程序。在Python中,我们通过标准库中的subprocess包来fork一个子进程,并运行一个外部的程序。
subprocess包中定义有数个创建子进程的函数,这些函数分别以不同的方式创建子进程,所以我们可以根据需要来从中选取一个使用。另外subprocess还提供了一些管理标准流(standard stream)和管道(pipe)的工具,从而在进程间使用文本通信。

subprocess.call()
父进程等待子进程完成
返回退出信息(returncode,相当于Linux exit code)

subprocess.check_call()
父进程等待子进程完成
返回0
检查退出信息,如果returncode不为0,则举出错误subprocess.CalledProcessError,该对象包含有returncode属性,可用try…except…来检查

subprocess.check_output()
父进程等待子进程完成
返回子进程向标准输出的输出结果
检查退出信息,如果returncode不为0,则举出错误subprocess.CalledProcessError,该对象包含有returncode属性和output属性,output属性为标准输出的输出结果,可用try…except…来检查。

这三个函数的使用方法相类似,下面来以subprocess.call()举例说明:

>>> import subprocess
>>> retcode = subprocess.call(["ls", "-l"])
#和shell中命令ls -a显示结果一样
>>> print retcode
0

将程序名(ls)和所带的参数(-l)一起放在一个表中传递给subprocess.call()

shell默认为False,在Linux下,shell=False时, Popen调用os.execvp()执行args指定的程序;shell=True时,如果args是字符串,Popen直接调用系统的Shell来执行args指定的程序,如果args是一个序列,则args的第一项是定义程序命令字符串,其它项是调用系统Shell时的附加参数。

上面例子也可以写成如下:

>>> retcode = subprocess.call("ls -l",shell=True)

在Windows下,不论shell的值如何,Popen调用CreateProcess()执行args指定的外部程序。如果args是一个序列,则先用list2cmdline()转化为字符串,但需要注意的是,并不是MS Windows下所有的程序都可以用list2cmdline来转化为命令行字符串。

subprocess.Popen()

class 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)

实际上,上面的几个函数都是基于Popen()的封装(wrapper)。这些封装的目的在于让我们容易使用子进程。当我们想要更个性化我们的需求的时候,就要转向Popen类,该类生成的对象用来代表子进程。

与上面的封装不同,Popen对象创建后,主程序不会自动等待子进程完成。我们必须调用对象的wait()方法,父进程才会等待 (也就是阻塞block),举例:

>>> import subprocess
>>> child = subprocess.Popen(['ping','-c','4','blog.linuxeye.com'])
>>> print 'parent process'

从运行结果中看到,父进程在开启子进程之后并没有等待child的完成,而是直接运行print。

对比等待的情况:

>>> import subprocess
>>> child = subprocess.Popen('ping -c4 blog.linuxeye.com',shell=True)
>>> child.wait()
>>> print 'parent process'

从运行结果中看到,父进程在开启子进程之后并等待child的完成后,再运行print。
此外,你还可以在父进程中对子进程进行其它操作,比如我们上面例子中的child对象:
child.poll() # 检查子进程状态
child.kill() # 终止子进程
child.send_signal() # 向子进程发送信号
child.terminate() # 终止子进程

子进程的PID存储在child.pid
二、子进程的文本流控制
子进程的标准输入、标准输出和标准错误如下属性分别表示:
child.stdin
child.stdout
child.stderr

可以在Popen()建立子进程的时候改变标准输入、标准输出和标准错误,并可以利用subprocess.PIPE将多个子进程的输入和输出连接在一起,构成管道(pipe),如下2个例子:
>>> import subprocess
>>> child1 = subprocess.Popen(["ls","-l"], stdout=subprocess.PIPE)
>>> print child1.stdout.read(),
#或者child1.communicate()
>>> import subprocess
>>> child1 = subprocess.Popen(["cat","/etc/passwd"], stdout=subprocess.PIPE)
>>> child2 = subprocess.Popen(["grep","0:0"],stdin=child1.stdout, stdout=subprocess.PIPE)
>>> out = child2.communicate()

subprocess.PIPE实际上为文本流提供一个缓存区。child1的stdout将文本输出到缓存区,随后child2的stdin从该PIPE中将文本读取走。child2的输出文本也被存放在PIPE中,直到communicate()方法从PIPE中读取出PIPE中的文本。
注意:communicate()是Popen对象的一个方法,该方法会阻塞父进程,直到子进程完成
Python 相关文章推荐
python2.7删除文件夹和删除文件代码实例
Dec 18 Python
使用Python编写一个在Linux下实现截图分享的脚本的教程
Apr 24 Python
Python正则表达式完全指南
May 25 Python
Python验证文件是否可读写代码分享
Dec 11 Python
python执行CMD指令,并获取返回的方法
Dec 19 Python
使用Python和Scribus创建一个RGB立方体的方法
Jul 17 Python
在Python中预先初始化列表内容和长度的实现
Nov 28 Python
python 输出列表元素实例(以空格/逗号为分隔符)
Dec 25 Python
python批量处理txt文件的实例代码
Jan 13 Python
Django中如何用xlwt生成表格的方法步骤
Jan 31 Python
Python基于爬虫实现全网搜索并下载音乐
Feb 14 Python
Python实现简单的2048小游戏
Mar 01 Python
用Python和MD5实现网站挂马检测程序
Mar 13 #Python
python操作数据库之sqlite3打开数据库、删除、修改示例
Mar 13 #Python
使用Python判断IP地址合法性的方法实例
Mar 13 #Python
Python 分析Nginx访问日志并保存到MySQL数据库实例
Mar 13 #Python
详解Python中的__init__和__new__
Mar 12 #Python
python文件和目录操作方法大全(含实例)
Mar 12 #Python
Python 文件读写操作实例详解
Mar 12 #Python
You might like
cakephp常见知识点汇总
2017/02/24 PHP
jquery 查找select ,并触发事件的实现代码
2011/03/30 Javascript
Jquery EasyUI的添加,修改,删除,查询等基本操作介绍
2013/10/11 Javascript
JavaScript中的数值范围介绍
2014/12/29 Javascript
DOM节点删除函数removeChild()用法实例
2015/01/12 Javascript
JavaScript监听和禁用浏览器回车事件实例
2015/01/31 Javascript
实例详解jQuery表单验证插件validate
2016/01/18 Javascript
JavaScript中的this引用(推荐)
2016/08/05 Javascript
关于动态生成dom绑定事件失效的原因及解决方法
2016/08/06 Javascript
基于JS+Canves实现点击按钮水波纹效果
2016/09/15 Javascript
详解vue 配合vue-resource调用接口获取数据
2017/06/22 Javascript
Vue中遍历数组的新方法实例详解
2019/07/21 Javascript
微信小程序自定义支持图片的弹窗
2020/12/21 Javascript
[48:23]DOTA2上海特级锦标赛主赛事日 - 4 败者组第四轮#1COL VS EG第一局
2016/03/05 DOTA
12步入门Python中的decorator装饰器使用方法
2016/06/20 Python
python实现汉诺塔算法
2021/03/01 Python
python面向对象法实现图书管理系统
2019/04/19 Python
pyqt5 删除layout中的所有widget方法
2019/06/25 Python
Python (Win)readline和tab补全的安装方法
2019/08/27 Python
Python实现Restful API的例子
2019/08/31 Python
Python使用指定字符长度切分数据示例
2019/12/05 Python
django框架基于queryset和双下划线的跨表查询操作详解
2019/12/11 Python
Python如何爬取51cto数据并存入MySQL
2020/08/25 Python
Python通过fnmatch模块实现文件名匹配
2020/09/30 Python
css3通过scale()、rotate()实现放大、旋转
2020/03/19 HTML / CSS
video下autoplay属性无效的解决方法(添加muted属性)
2020/05/19 HTML / CSS
铭宣海淘转运:美国、日本、英国转运等全球转运公司
2019/09/10 全球购物
我能否用void** 指针作为参数, 使函数按引用接受一般指针
2013/02/16 面试题
临床医师专业个人自我评价范文
2013/11/07 职场文书
公司门卫的岗位职责
2014/02/19 职场文书
葬礼司仪主持词
2014/03/31 职场文书
小学教育见习报告
2014/10/31 职场文书
2015年考研复习计划
2015/01/19 职场文书
大学学生会主席竞选稿
2015/11/19 职场文书
Python保存并浏览用户的历史记录
2022/04/29 Python
flex布局中使用flex-wrap实现换行的项目实践
2022/06/21 HTML / CSS