如何使用python的subprocess执行命令、交互、等待、是否结束及解析JSON结果

 更新时间:2023年12月22日 16:14:19   作者:小龙在山东  
这篇文章主要给大家介绍了关于如何使用python的subprocess执行命令、交互、等待、是否结束及解析JSON结果的相关资料,subprocess模块提供了一种简单的方法来创建和管理子进程,它可以让我们在Python程序中执行外部命令,获取命令的输出和错误信息,需要的朋友可以参考下

前言

Python的subprocess模块提供了一种在Python中调用外部命令的方法。它允许您在Python程序中启动新进程,连接到它们的输入/输出/错误管道,并等待它们完成。

常用用法

下面是一些subprocess模块的常用用法:

  • 运行外部命令并获取输出:
import subprocess

output = subprocess.check_output(["ls", "-l"])
print(output.decode())
  • 运行外部命令并获取返回值:
import subprocess

return_code = subprocess.call(["ls", "-l"])
print(return_code)
  • 运行外部命令并将输出重定向到文件:
import subprocess

with open("output.txt", "w") as f:
    subprocess.call(["ls", "-l"], stdout=f)
  • 运行外部命令并将输入从文件中读取:
import subprocess

with open("input.txt", "r") as f:
    subprocess.call(["grep", "hello"], stdin=f)
  • 运行外部命令并将输入从Python程序中提供:
import subprocess

subprocess.call(["grep", "hello"], input=b"hello world
")
  • 运行外部命令并捕获标准错误:
import subprocess

try:
    subprocess.check_output(["ls", "-l", "/nonexistent"])
except subprocess.CalledProcessError as e:
    print(e.stderr.decode())
  • 运行外部命令并等待它完成:
import subprocess

p = subprocess.Popen(["sleep", "5"])
p.wait()
print("Done")

以上是subprocess模块的一些常用用法,更多详细信息请参考Python官方文档。

创建一个新的进程

Python的subprocess模块中的Popen函数用于创建一个新的进程,并与其进行交互。Popen函数的语法如下:

subprocess.Popen(args, bufsize=-1, executable=None, stdin=None, stdout=None, stderr=None, 
preexec_fn=None, close_fds=True, shell=False, cwd=None, env=None, 
universal_newlines=False, startupinfo=None, creationflags=0)

参数说明:

  • args:要执行的命令,可以是一个字符串或一个列表。如果是一个字符串,则会被解释为一个shell命令;如果是一个列表,则第一个元素是要执行的命令,后面的元素是命令的参数。
  • bufsize:缓冲区大小,默认为-1,表示使用系统默认值。
  • executable:要执行的可执行文件的路径,默认为None,表示使用系统默认的可执行文件。
  • stdin、stdout、stderr:分别表示标准输入、标准输出、标准错误输出的文件描述符。默认为None,表示使用父进程的标准输入、标准输出、标准错误输出。
  • preexec_fn:在子进程执行前被调用的可执行对象,可以是一个函数或一个可调用对象。默认为None。
  • close_fds:如果为True,则在子进程中关闭所有文件描述符。默认为True。
  • shell:如果为True,则将args作为一个shell命令执行。默认为False。
  • cwd:子进程的当前工作目录。默认为None,表示使用父进程的当前工作目录。
  • env:子进程的环境变量。默认为None,表示使用父进程的环境变量。
  • universal_newlines:如果为True,则将stdin、stdout、stderr的数据以文本模式处理。默认为False。
  • startupinfo:用于指定子进程的一些启动信息,如窗口大小、标题等。默认为None。
  • creationflags:用于指定子进程的一些标志,如CREATE_NEW_CONSOLE、CREATE_NEW_PROCESS_GROUP等。默认为0。

Popen函数返回一个Popen对象,可以通过该对象的方法和属性与子进程进行交互,如:

  • communicate(input=None, timeout=None):与子进程进行交互,发送input数据并等待子进程执行完毕。如果timeout不为None,则在指定时间内等待子进程执行完毕。
  • poll():检查子进程是否已经结束,如果已经结束则返回子进程的退出状态码,否则返回None。
  • wait(timeout=None):等待子进程执行完毕,并返回子进程的退出状态码。如果timeout不为None,则在指定时间内等待子进程执行完毕。
  • send_signal(signal):向子进程发送信号。
  • terminate():向子进程发送SIGTERM信号,终止子进程。
  • kill():向子进程发送SIGKILL信号,强制终止子进程。
  • pid:子进程的进程ID。
  • returncode:子进程的退出状态码。

示例代码:

import subprocess

# 执行一个简单的命令
p = subprocess.Popen('ls -l', shell=True)
p.wait()

# 执行一个带参数的命令
p = subprocess.Popen(['ls', '-l'], stdout=subprocess.PIPE)
output, error = p.communicate()
print(output.decode())

# 执行一个长时间运行的命令,并在指定时间内等待其执行完毕
p = subprocess.Popen('sleep 10', shell=True)
try:
    p.wait(timeout=5)
except subprocess.TimeoutExpired:
    p.kill()

与子进程进行交互

communicate()subprocess模块中的一个方法,用于与子进程进行交互。它会向子进程的标准输入发送数据,并等待子进程完成任务后获取其标准输出和标准错误输出。

communicate()方法的语法如下:

stdout, stderr = subprocess.communicate(input=None, timeout=None)

其中,input参数是要发送给子进程的数据,可以是字符串或字节流。如果不需要向子进程发送数据,则可以将其设置为Nonetimeout参数是等待子进程完成任务的超时时间,单位为秒。如果子进程在超时时间内未完成任务,则会抛出TimeoutExpired异常。

communicate()方法会返回一个元组,其中第一个元素是子进程的标准输出,第二个元素是子进程的标准错误输出。如果子进程没有输出,则对应的元素为None

下面是一个使用communicate()方法的示例:

import subprocess

# 执行命令
p = subprocess.Popen(['ls', '-l'], stdout=subprocess.PIPE, stderr=subprocess.PIPE)

# 获取子进程的输出
stdout, stderr = p.communicate()

# 输出子进程的标准输出和标准错误输出
print(stdout.decode('utf-8'))
print(stderr.decode('utf-8'))

在上面的示例中,我们使用Popen()方法创建了一个子进程,并将其标准输出和标准错误输出分别重定向到管道中。然后,我们使用communicate()方法等待子进程完成任务,并获取其标准输出和标准错误输出。最后,我们将其转换为字符串并输出。

执行结果解析成json格式

可以使用Python的json模块将subprocess执行结果解析成json格式。

假设subprocess执行的命令是获取系统信息的命令,如下所示:

import subprocess

cmd = "systeminfo"
result = subprocess.run(cmd, stdout=subprocess.PIPE, stderr=subprocess.PIPE, shell=True)

执行结果保存在result变量中,可以使用json模块将其解析成json格式,如下所示:

import json

output = result.stdout.decode('utf-8')
json_output = json.loads(output)

其中,result.stdout是subprocess执行结果的标准输出,使用decode方法将其转换成字符串类型。然后使用json.loads方法将字符串解析成json格式。

解析后的json格式可以按照需要进行处理和使用。

检查子进程是否已经结束

在subprocess中,poll()方法用于检查子进程是否已经结束。如果子进程已经结束,poll()方法会返回子进程的退出状态码。如果子进程还在运行,poll()方法会返回None。

下面是一个使用poll()方法的示例:

import subprocess

# 启动子进程
p = subprocess.Popen(['ls', '-l'])

# 检查子进程是否已经结束
while p.poll() is None:
    print('子进程还在运行...')
    
# 子进程已经结束,获取退出状态码
print('子进程已经结束,退出状态码为:', p.returncode)

在上面的示例中,我们启动了一个子进程来执行ls命令,然后使用poll()方法检查子进程是否已经结束。如果子进程还在运行,就会一直输出“子进程还在运行…”,直到子进程结束。当子进程结束后,我们使用returncode属性获取子进程的退出状态码,并输出到控制台。

等待子进程结束

在subprocess模块中,wait()方法用于等待子进程结束并返回状态码。它会阻塞当前进程,直到子进程结束为止。

wait()方法的语法如下:

status = subprocess.Popen.wait(self, timeout=None, endtime=None)

其中,timeout参数表示等待子进程结束的最长时间,单位为秒;endtime参数表示等待子进程结束的最晚时间,是一个时间戳。

如果子进程已经结束,wait()方法会立即返回状态码;如果子进程还在运行,wait()方法会阻塞当前进程,直到子进程结束为止。

wait()方法返回的状态码是一个整数,表示子进程的退出状态。如果子进程正常结束,状态码为0;如果子进程异常结束,状态码为一个非零值,具体的值表示异常的类型。

判断是否执行成功

在Python中,可以使用subprocess模块的returncode属性来判断子进程是否执行成功。如果子进程成功执行,returncode属性的值为0;如果子进程执行失败,returncode属性的值为非零。可以通过以下代码来实现:

import subprocess

# 执行命令
result = subprocess.run(['ls', '-l'], stdout=subprocess.PIPE, stderr=subprocess.PIPE)

# 判断是否执行成功
if result.returncode == 0:
    print('执行成功')
else:
    print('执行失败')

在上面的代码中,我们使用subprocess.run()方法执行了一个ls -l命令,并将结果保存在result变量中。然后,我们通过判断result.returncode的值来判断子进程是否执行成功。如果returncode的值为0,则表示执行成功;否则,表示执行失败。

总结

到此这篇关于如何使用python的subprocess执行命令、交互、等待、是否结束及解析JSON结果的文章就介绍到这了,更多相关python subprocess模块用法内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • Pandas中DataFrame的分组/分割/合并的实现

    Pandas中DataFrame的分组/分割/合并的实现

    这篇文章主要介绍了Pandas中DataFrame的分组/分割/合并的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2019-07-07
  • Python中 map()函数的用法详解

    Python中 map()函数的用法详解

    map( )函数在算法题目里面经常出现,map( )会根据提供的函数对指定序列做映射,在写返回值等需要转换的时候比较常用。这篇文章主要介绍了Python中 map()的用法,需要的朋友可以参考下
    2018-07-07
  • python自动重试第三方包retrying模块的方法

    python自动重试第三方包retrying模块的方法

    retrying是一个python的重试包,可以用来自动重试一些可能运行失败的程序段。这篇文章主要介绍了python自动重试第三方包retrying的方法,需要的朋友参考下吧
    2018-04-04
  • Django教程笔记之中间件middleware详解

    Django教程笔记之中间件middleware详解

    Django 中的中间件(middleware),是一个镶嵌到Django的request/response处理机制中的一个hooks框架,是一个修改django全局输入输出的一个底层插件系统。这篇文章主要给大家介绍了关于Django教程笔记之中间件middleware的相关资料,需要的朋友可以参考下
    2018-08-08
  • 爬山算法简介和Python实现实例

    爬山算法简介和Python实现实例

    这篇文章主要介绍了爬山算法,爬山法(climbing method)是一种优化算法,其一般从一个随机的解开始,然后逐步找到一个最优解(局部最优)然后用Python实现了这个算法,需要的朋友可以参考下
    2014-04-04
  • python GUI库图形界面开发之PyQt5浏览器控件QWebEngineView详细使用方法

    python GUI库图形界面开发之PyQt5浏览器控件QWebEngineView详细使用方法

    这篇文章主要介绍了python GUI库图形界面开发之PyQt5浏览器控件QWebEngineView详细使用方法,需要的朋友可以参考下
    2020-02-02
  • Python实现双轴组合图表柱状图和折线图的具体流程

    Python实现双轴组合图表柱状图和折线图的具体流程

    这篇文章主要介绍了Python双轴组合图表柱状图+折线图,Python绘制双轴组合的关键在plt库的twinx()函数,具体实例代码跟随小编一起看看吧
    2021-08-08
  • Python自定义一个异常类的方法

    Python自定义一个异常类的方法

    在本篇文章里小编给大家分享了关于Python自定义一个异常类的详细步骤和实例代码,有兴趣的朋友们参考学习下。
    2019-06-06
  • 35个Python编程小技巧

    35个Python编程小技巧

    从我开始学习python的时候,我就开始自己总结一个python小技巧的集合。后来当我什么时候在Stack Overflow或者在某个开源软件里看到一段很酷代码的时候,我就很惊讶:原来还能这么做!,当时我会努力的自己尝试一下这段代码,直到我懂了它的整体思路以后,我就把这段代码加到我的集合里
    2014-04-04
  • Python中的文件和目录操作实现代码

    Python中的文件和目录操作实现代码

    对于文件和目录的处理,虽然可以通过操作系统命令来完成,但是Python语言为了便于开发人员以编程的方式处理相关工作,提供了许多处理文件和目录的内置函数。重要的是,这些函数无论是在Unix、Windows还是Macintosh平台上,它们的使用方式是完全一致的。
    2011-03-03

最新评论