在Python循环期间设置超时?

时间:2015-10-17 22:51:28

标签: python loops timeout subprocess

所以我在一个目录中运行一个循环,在该目录中,由subprocess.Popen运行的外部脚本正在通过目录移动并对每个文件执行计算。外部脚本有点不稳定,当它遇到不知道如何处理的文件时偶尔会冻结。有没有办法为subprocess.Popen添加超时功能,以便我可以跳过该文件继续下一步?

编辑: 这是我的循环:

def automate():
    os.chdir("/home/mlts/dir")
    working_dir = b"/home/mts/dir"
    for filename in os.listdir(working_dir):
        if filename.endswith(".MTS"):
            try:
                print("Performing calculations on {filename!r}...".format(**vars()))
                try:
                    os.remove("mts_tbl.txt")
                except OSError:
                    pass
                time.sleep(3)
                p = Popen(["run_command.command", "-f", "a"], cwd=working_dir, stdin=PIPE)
                p.communicate(input=b"\n".join([b"1", str(filename), b"8", b"alloy-liquid", b"0", b"x", b"5", b"4", b"-1.6", b"4", b"1", b"0"]))

1 个答案:

答案 0 :(得分:0)

要在Python 2上强制timeout(stdlib中不存在.communicate(timeout=)),您可以使用threading.Timer()

p = Popen(...)
t = Timer(3, p.kill) # kill in 3 seconds
t.start()
p.communicate(...)
t.cancel() # no need to kill, the process is dead already

完整的例子:

import os
import traceback
from glob import glob
from subprocess import Popen, PIPE
from threading import Timer

os.chdir("/home/mls/dir") #XXX assume mlts is a typo in the question
for filename in glob(b"*.MTS"):
    print("Performing calculations on {filename!r}...".format(**vars()))
    try:
        os.remove("mts_tbl.txt")
    except OSError:
        pass # ignore
    try:
        p = Popen(["run_command.command", "-f", "a"], stdin=PIPE) #XXX no cwd
        t = Timer(3, p.kill)
        t.start()
        p.communicate(input=b"\n".join([b"1", filename,
                b"8\nalloy-liquid\n0\nx\n5\n4\n-1.6\n4\n1\n0"]))
        t.cancel()
    except Exception:
        traceback.print_exc()

此代码假定您要在父Python脚本的当前工作目录中运行子进程。

在Python 3上或者如果安装了subprocess32;您可以将timeout参数传递给.communicate()方法,而不是使用Timer()