python模块之subprocess模块级方法的使用

作者:当麻的小红箱 时间:2022-05-10 03:28:32 

subprocess.run()

运行并等待args参数指定的指令完成,返回CompletedProcess实例。

参数:(*popenargs, input=None, capture_output=False, timeout=None, check=False, **kwargs)。除input, capture_output, timeout, check,其他参数与Popen构造器参数一致。

capture_output:如果设置为True,表示重定向stdout和stderr到管道,且不能再传递stderr或stdout参数,否则抛出异常。

input:input参数将作为子进程的标准输入传递给Popen.communicate()方法,必须是string(需要指定encoding或errors参数,或者设置text为True)或byte类型。非None的input参数不能和stdin参数一起使用,否则将抛出异常,构造Popen实例的stdin参数将指定为subprocess.PIPE。

timeout:传递给Popen.communicate()方法。

check:如果设置为True,进程执行返回非0状态码将抛出CalledProcessError异常。


# 源码

def run(*popenargs, input=None, capture_output=False, timeout=None, check=False, **kwargs):
 if input is not None:
   if 'stdin' in kwargs:
     raise ValueError('stdin and input arguments may not both be used.')
   kwargs['stdin'] = PIPE

if capture_output:
   if ('stdout' in kwargs) or ('stderr' in kwargs):
     raise ValueError('stdout and stderr arguments may not be used '
              'with capture_output.')
   kwargs['stdout'] = PIPE
   kwargs['stderr'] = PIPE

with Popen(*popenargs, **kwargs) as process:
   try:
     stdout, stderr = process.communicate(input, timeout=timeout)
   except TimeoutExpired:
     process.kill()
     stdout, stderr = process.communicate()
     raise TimeoutExpired(process.args, timeout, output=stdout,
                stderr=stderr)
   except: # Including KeyboardInterrupt, communicate handled that.
     process.kill()
     # We don't call process.wait() as .__exit__ does that for us.
     raise
   retcode = process.poll()
   if check and retcode:
     raise CalledProcessError(retcode, process.args,
                  output=stdout, stderr=stderr)
 return CompletedProcess(process.args, retcode, stdout, stderr)

python3.5版本前,call(), check_all(), checkoutput()三种方法构成了subprocess模块的高级API。

subprocess.call()

运行并等待args参数指定的指令完成,返回执行状态码(Popen实例的returncode属性)。

参数:(*popenargs, timeout=None, **kwargs)。与Popen构造器参数基本相同,除timeout外的所有参数都将传递给Popen接口。

调用call()函数不要使用stdout=PIPE或stderr=PIPE,因为如果子进程生成了足量的输出到管道填满OS管道缓冲区,子进程将因不能从管道读取数据而导致阻塞。


# 源码
def call(*popenargs, timeout=None, **kwargs):
 with Popen(*popenargs, **kwargs) as p:
   try:
     return p.wait(timeout=timeout)
   except:
     p.kill()
     p.wait()
     raise

subprocess.check_call()

运行并等待args参数指定的指令完成,返回0状态码或抛出CalledProcessError异常,该异常的cmd和returncode属性可以查看执行异常的指令和状态码。

参数:(*popenargs, **kwargs)。全部参数传递给call()函数。

注意事项同call()


# 源码
def check_call(*popenargs, **kwargs):
 retcode = call(*popenargs, **kwargs)
 if retcode:
   cmd = kwargs.get("args")
   if cmd is None:
     cmd = popenargs[0]
   raise CalledProcessError(retcode, cmd)
 return 0

subprocess.check_output()

运行并等待args参数指定的指令完成,返回标准输出(CompletedProcess实例的stdout属性),类型默认是byte字节,字节编码可能取决于执行的指令,设置universal_newlines=True可以返回string类型的值。
如果执行状态码非0,将抛出CalledProcessError异常。

参数:(*popenargs, timeout=None, **kwargs)。全部参数传递给run()函数,但不支持显示地传递input=None继承父进程的标准输入文件句柄。

要在返回值中捕获标准错误,设置stderr=subprocess.STDOUT;也可以将标准错误重定向到管道stderr=subprocess.PIPE,通过CalledProcessError异常的stderr属性访问。


# 源码

def check_output(*popenargs, timeout=None, **kwargs):
 if 'stdout' in kwargs:
   raise ValueError('stdout argument not allowed, it will be overridden.')

if 'input' in kwargs and kwargs['input'] is None:
   # Explicitly passing input=None was previously equivalent to passing an
   # empty string. That is maintained here for backwards compatibility.
   kwargs['input'] = '' if kwargs.get('universal_newlines', False) else b''

return run(*popenargs, stdout=PIPE, timeout=timeout, check=True,
       **kwargs).stdout

subprocess模块还提供了python2.x版本中commands模块的相关函数。

subprocess.getstatusoutput(cmd)

实际上是调用check_output()函数,在shell中执行string类型的cmd指令,返回(exitcode, output)形式的元组,output(包含stderrstdout)是使用locale encoding解码的字符串,并删除了结尾的换行符。


# 源码
try:
 data = check_output(cmd, shell=True, universal_newlines=True, stderr=STDOUT)
 exitcode = 0
except CalledProcessError as ex:
 data = ex.output
 exitcode = ex.returncode
if data[-1:] == '\n':
 data = data[:-1]
return exitcode, data

subprocess.getoutput(cmd)

getstatusoutput()类似,但结果只返回output。

来源:https://segmentfault.com/a/1190000018658746

标签:python,subprocess
0
投稿

猜你喜欢

  • 在Oracle PL/SQL中游标声明中表名动态变化的方法

    2009-02-28 10:39:00
  • thinkphp3.2.3版本的数据库增删改查实现代码

    2023-10-19 15:07:53
  • block 和 inline 的区别是?

    2009-12-08 13:00:00
  • asp检测服务器XmlHttp组件支持情况

    2008-03-03 12:30:00
  • 导航与搜索合并的可能性

    2009-09-27 12:06:00
  • VML模拟Rose画流程图代码

    2008-09-02 10:58:00
  • 可以在线创建文件夹吗?

    2009-11-01 18:07:00
  • 强大的Perl正则表达式实例详解

    2023-08-09 11:12:33
  • MySQL查询优化

    2009-03-09 14:41:00
  • 如何尽快释放掉Connection对象建立的连接?

    2009-12-16 18:38:00
  • 根据表名和索引获取需要的列名的存储过程

    2011-09-30 11:54:42
  • python random从集合中随机选择元素的方法

    2023-07-05 08:29:45
  • laravel中短信发送验证码的实现方法

    2023-11-15 08:05:16
  • 用DIV+CSS编写出的网页中的圆角矩形

    2007-10-14 16:03:00
  • asp如何在线更改密码?

    2010-06-26 12:22:00
  • 从算法入手讲解SQL Server的典型示例

    2008-12-18 14:51:00
  • SQL提高查询效率之Like篇

    2011-10-01 09:36:42
  • SQL SERVER数据库开发之asp存储过程应用

    2008-05-19 12:55:00
  • 谷歌历年母亲节Logo一览

    2009-05-11 12:17:00
  • Web 前端优化最佳实践之 Cookie 篇

    2008-06-25 13:00:00
  • asp之家 网络编程 m.aspxhome.com