Python 创建子进程模块subprocess详解


Posted in Python onApril 08, 2015

最近,我们老大要我写一个守护者程序,对服务器进程进行守护。如果服务器不幸挂掉了,守护者能即时的重启应用程序。上网Google了一下,发现Python有很几个模块都可以创建进程。最终我选择使用subprocess模块,因为在Python手册中有这样一段话:

This module intends to replace several other, older modules and functions, such as: os.system、os.spawn*、os.popen*、popen2.*、commands.*

subprocess被用来替换一些老的模块和函数,如:os.system、os.spawn*、os.popen*、popen2.*、commands.*。可见,subprocess是被推荐使用的模块。

下面是一个很简单的例子,创建一个新进程,执行app1.exe,传入相当的参数,并打印出进程的返回值:

import subprocess
returnCode = subprocess.call('app1.exe -a -b -c -d')

print 'returncode:', returnCode
#----- 结果 --------

#Python is powerful

#app1.exe

#-a

#-b

#-c

#-d

returncode: 0

app1.exe是一个非常简单的控制台程序,它只打印出传入的参数,代码如下:
#include <iostream>

using namespace std;
int main(int argc, const char *argv[])

{

    cout << "Python is powerful" << endl;

    for (int i = 0; i < argc; i++)

    {

        cout << argv[i] << endl;

    }
    return 0;

}

 闲话少说,下面开始详细介绍subprocess模块。subprocess模块中只定义了一个类: Popen。可以使用Popen来创建进程,并与进程进行复杂的交互。它的构造函数如下:

subprocess.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)

参数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()函数,用于设置子进程的一些属性,如:主窗口的外观,进程的优先级等等。

subprocess.PIPE

在创建Popen对象时,subprocess.PIPE可以初始化stdin, stdout或stderr参数。表示与子进程通信的标准流。

subprocess.STDOUT

创建Popen对象时,用于初始化stderr参数,表示将错误通过标准输出流输出。

Popen的方法:
Popen.poll()

用于检查子进程是否已经结束。设置并返回returncode属性。
Popen.wait()

等待子进程结束。设置并返回returncode属性。
Popen.communicate(input=None)

与子进程进行交互。向stdin发送数据,或从stdout和stderr中读取数据。可选参数input指定发送到子进程的参数。Communicate()返回一个元组:(stdoutdata, stderrdata)。注意:如果希望通过进程的stdin向其发送数据,在创建Popen对象的时候,参数stdin必须被设置为PIPE。同样,如果希望从stdout和stderr获取数据,必须将stdout和stderr设置为PIPE。
Popen.send_signal(signal)

向子进程发送信号。
Popen.terminate()

停止(stop)子进程。在windows平台下,该方法将调用Windows API TerminateProcess()来结束子进程。
Popen.kill()

杀死子进程。
Popen.stdin

如果在创建Popen对象是,参数stdin被设置为PIPE,Popen.stdin将返回一个文件对象用于策子进程发送指令。否则返回None。
Popen.stdout

如果在创建Popen对象是,参数stdout被设置为PIPE,Popen.stdout将返回一个文件对象用于策子进程发送指令。否则返回None。
Popen.stderr

如果在创建Popen对象是,参数stdout被设置为PIPE,Popen.stdout将返回一个文件对象用于策子进程发送指令。否则返回None。
Popen.pid

获取子进程的进程ID。
Popen.returncode

获取进程的返回值。如果进程还没有结束,返回None。
 下面是一个非常简单的例子,来演示supprocess模块如何与一个控件台应用程序进行交互。

 

 import subprocess
p = subprocess.Popen("app2.exe", stdin = subprocess.PIPE, /

    stdout = subprocess.PIPE, stderr = subprocess.PIPE, shell = False)
p.stdin.write('3/n')

p.stdin.write('4/n')

print p.stdout.read()
#---- 结果 ----

input x: 

input y: 

3 + 4 = 7

 

 app2.exe也是一个非常简单的控制台程序,它从界面上接收两个数值,执行加操作,并将结果打印到控制台上。代码如下:
 

 #include <iostream>

using namespace std;
int main(int argc, const char *artv[])

{

    int x, y;

    cout << "input x: " << endl;

    cin >> x;

    cout << "input y: " << endl;

    cin >> y;

    cout << x << " + " << y << " = " << x + y << endl;
    return 0;

}

 

  supprocess模块提供了一些函数,方便我们用于创建进程。
subprocess.call(*popenargs, **kwargs)

运行命令。该函数将一直等待到子进程运行结束,并返回进程的returncode。文章一开始的例子就演示了call函数。如果子进程不需要进行交互,就可以使用该函数来创建。
subprocess.check_call(*popenargs, **kwargs)

与subprocess.call(*popenargs, **kwargs)功能一样,只是如果子进程返回的returncode不为0的话,将触发CalledProcessError异常。在异常对象中,包括进程的returncode信息。
 

subprocess模块的内容就这么多。在Python手册中,还介绍了如何使用subprocess来替换一些老的模块,老的函数的例子。赶兴趣的朋友可以看一下。

Python 相关文章推荐
Python3实现连接SQLite数据库的方法
Aug 23 Python
Python的Django应用程序解决AJAX跨域访问问题的方法
May 31 Python
python解决方案:WindowsError: [Error 2]
Aug 28 Python
详解python3实现的web端json通信协议
Dec 29 Python
用Python写一段用户登录的程序代码
Apr 22 Python
2019 Python最新面试题及答案16道题
Apr 11 Python
python中时间模块的基本使用教程
May 14 Python
Python使用py2neo操作图数据库neo4j的方法详解
Jan 13 Python
Python sublime安装及配置过程详解
Jun 29 Python
详解利用python识别图片中的条码(pyzbar)及条码图片矫正和增强
Nov 17 Python
Python+MySQL随机试卷及答案生成程序的示例代码
Feb 01 Python
python中pymysql包操作数据库方法
Apr 19 Python
Python中使用gzip模块压缩文件的简单教程
Apr 08 #Python
Python使用scrapy抓取网站sitemap信息的方法
Apr 08 #Python
Python中用format函数格式化字符串的用法
Apr 08 #Python
简单介绍Python中的JSON模块
Apr 08 #Python
Python实现把xml或xsl转换为html格式
Apr 08 #Python
Python正则表达式匹配HTML页面编码
Apr 08 #Python
在Python中关于中文编码问题的处理建议
Apr 08 #Python
You might like
2020年4月放送!《Princess Connect Re:Dive》制作组 & 角色声优公开!
2020/03/06 日漫
用PHP生成自己的LOG文件
2006/10/09 PHP
php array的学习笔记
2012/05/10 PHP
Android ProgressBar进度条和ProgressDialog进度框的展示DEMO
2013/06/19 PHP
JavaScript加密解密7种方法总结分析
2007/10/07 Javascript
IE和Firefox下javascript的兼容写法小结
2008/12/10 Javascript
jquery一句话全选/取消全选
2011/03/01 Javascript
一个分享按钮的插件使用介绍(可扩展,内附开发制作流程)
2011/09/19 Javascript
js控制CSS样式属性语法对照表
2012/12/11 Javascript
jQuery点击tr实现checkbox选中的方法
2013/03/19 Javascript
js动态往表格的td中添加图片并注册事件
2014/06/12 Javascript
node.js中的path.join方法使用说明
2014/12/08 Javascript
js判断某个方法是否存在实例代码
2015/01/10 Javascript
JQuery判断radio(单选框)是否选中和获取选中值方法总结
2015/04/15 Javascript
jquery控制页面部分刷新的方法
2015/06/24 Javascript
详解Node.js中的事件机制
2016/09/22 Javascript
jquery ui sortable拖拽后保存位置
2017/04/27 jQuery
BootStrap表单时间选择器详解
2017/05/09 Javascript
通过js控制时间,一秒一秒自己动的实例
2017/10/25 Javascript
ES6基础之解构赋值(destructuring assignment)
2019/02/21 Javascript
详解VScode编辑器vue环境搭建所遇问题解决方案
2019/04/26 Javascript
jquery中attr、prop、data区别与用法分析
2019/09/25 jQuery
vue动态加载SVG文件并修改节点数据的操作代码
2020/08/17 Javascript
python检测远程端口是否打开的方法
2015/03/14 Python
python实现的守护进程(Daemon)用法实例
2015/06/02 Python
为什么入门大数据选择Python而不是Java?
2018/03/07 Python
Python numpy.array()生成相同元素数组的示例
2018/11/12 Python
pycharm 解除默认unittest模式的方法
2018/11/30 Python
企业内部培训方案
2014/02/04 职场文书
《植物妈妈有办法》教学反思
2014/02/25 职场文书
民政局离婚协议书范本
2014/10/20 职场文书
2014年学生会工作总结
2014/11/07 职场文书
2015年度合同管理工作总结
2015/05/22 职场文书
党风廉洁教育心得体会
2016/01/20 职场文书
解决thinkphp6(tp6)在状态码500下不报错,或者显示错误“Malformed UTF-8 characters”的问题
2021/04/01 PHP
python如何正确使用yield
2021/05/21 Python