如何使用 python 2.7.6 使 subprocess.call 超时?

Posted

技术标签:

【中文标题】如何使用 python 2.7.6 使 subprocess.call 超时?【英文标题】:How to make a subprocess.call timeout using python 2.7.6? 【发布时间】:2014-06-10 20:20:22 【问题描述】:

这可能已经被问过了,但我在使用 python 2.7 时找不到任何关于 subprocess.call 超时的信息

【问题讨论】:

看看:***.com/questions/1191374/subprocess-with-timeout 【参考方案1】:

您可以使用subprocess32mentioned by @gps,它是 Python 3.2 - 3.5 的子进程标准库模块的反向移植,可用于 Python 2。

首先,安装 subprocess32 模块:

pip install subprocess32

这是一个代码sn-p:

>>> import subprocess32
>>> print subprocess32.check_output(["python", "--version"])
Python 2.7.12

>>> subprocess32.check_output(["sleep", "infinity"], timeout=3)
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
  File "/usr/local/lib/python2.7/dist-packages/subprocess32.py", line 340, in check_output
    raise TimeoutExpired(process.args, timeout, output=output)
subprocess32.TimeoutExpired: Command '['sleep', 'infinity']' timed out after 3 seconds

注意,默认timeout=None,表示永不超时。

【讨论】:

【参考方案2】:

我一直使用 2.7 超时的一种简单方法是使用 subprocess.poll()time.sleep() 延迟。这是一个非常基本的例子:

import subprocess
import time

x = #some amount of seconds
delay = 1.0
timeout = int(x / delay)

args = #a string or array of arguments
task = subprocess.Popen(args)

#while the process is still executing and we haven't timed-out yet
while task.poll() is None and timeout > 0:
     #do other things too if necessary e.g. print, check resources, etc.
     time.sleep(delay)
     timeout -= delay

如果您设置x = 600,那么您的超时将达到 10 分钟。而task.poll() 会查询进程是否已经终止。 time.sleep(delay) 在这种情况下会休眠 1 秒,然后将超时时间减少 1 秒。您可以随心所欲地玩弄这部分,但基本概念始终相同。

希望这会有所帮助!

subprocess.poll()https://docs.python.org/2/library/subprocess.html#popen-objects

【讨论】:

这不会终止进程。你需要添加 os.killpg(os.getpgid(task.pid), signal.SIGTERM) @AaronS 我认为task.terminate() 的作用几乎相同。【参考方案3】:

您可以安装 subprocess32 module mentioned by @gps -- Python 3.2/3.3 中 subprocess 模块的反向端口,用于 2.x。它适用于 Python 2.7,并且包含来自 Python 3.3 的超时支持。

subprocess.call() is just Popen().wait() 从而在timeout 秒内中断一个长时间运行的进程:

#!/usr/bin/env python
import time
from subprocess import Popen

p = Popen(*call_args)
time.sleep(timeout)
try:
    p.kill()
except OSError:
    pass # ignore
p.wait()

如果子进程可能更早结束,那么可移植的解决方案是use Timer() as suggested in @sussudio's answer:

#!/usr/bin/env python
from subprocess import Popen
from threading import Timer

def kill(p):
    try:
        p.kill()
    except OSError:
        pass # ignore

p = Popen(*call_args)
t = Timer(timeout, kill, [p])
t.start()
p.wait()
t.cancel()

在 Unix 上,你可以use SIGALRM as suggested in @Alex Martelli's answer:

#!/usr/bin/env python
import signal
from subprocess import Popen

class Alarm(Exception):
    pass

def alarm_handler(signum, frame):
    raise Alarm

signal.signal(signal.SIGALRM, alarm_handler)


p = Popen(*call_args)
signal.alarm(timeout)  # raise Alarm in 5 minutes
try:
    p.wait()
    signal.alarm(0)  # reset the alarm
except Alarm:
    p.kill()
    p.wait()

为避免在此处使用线程和信号,Python 3 上的subprocess 模块使用busy loop with waitpid(WNOHANG) calls on Unix 和winapi.WaitForSingleObject() on Windows。

【讨论】:

【参考方案4】:

您可以尝试使用“easyprocess”

https://github.com/ponty/EasyProcess

它有许多您需要的功能,例如“超时”

【讨论】:

【参考方案5】:

在 python 3.3 中添加了 timeout 参数。

https://docs.python.org/3/library/subprocess.html#subprocess.call

【讨论】:

感谢您的回复。我知道我们使用的是python 2.7.6

以上是关于如何使用 python 2.7.6 使 subprocess.call 超时?的主要内容,如果未能解决你的问题,请参考以下文章

Python:执行cmd命令时读取输出

带有 Python 2.7.6 和 Virtualenv 12.0.7 (OSX10,10.2) 的 Django 1.7.6 - 模板呈现模板语法错误

使用 EB CLI 3.4.5 (Python 2.7.6) 发布 ElasticBeanStalk/Tomcat 战争部署

使用virtualenv管理多个Python版本

python进阶学习chapter02(列表字典集合操作)

CentOS 6.X怎么更新Python2.7.x版本