如何在不检查标志的情况下终止Python线程

时间:2013-04-28 10:56:34

标签: python multithreading python-2.7 subprocess python-multithreading

class My_Thread(threading.Thread):

    def __init__(self):
        threading.Thread.__init__(self)

    def run(self):
        print "Starting " + self.name
        cmd = [ "bash", 'process.sh']
        p = subprocess.Popen(cmd,
                     stdout=subprocess.PIPE,
                     stderr=subprocess.STDOUT)
        for line in iter(p.stdout.readline, b''):
            print ("-- " + line.rstrip())
        print "Exiting " + self.name

    def stop(self):
        print "Trying to stop thread "
        self.run = False

thr = My_Thread()
thr.start()
time.sleep(30)
thr.stop()
thr.join()

所以我有如上所示的线程,实际上是在windows上工作而且process.sh是在cygwin中运行的bash脚本,需要大约5分钟才能完成执行所以它不是一个循环它的一些模拟proecess

我想在这个类中创建stop()函数,以便我可以在需要时立即终止脚本。终止后,我不希望process.sh脚本有任何有用的结果

请您建议任何方法,如果可能的话,请给出一点解释..

2 个答案:

答案 0 :(得分:9)

对于您的特定示例,通过终止使用Popen对象的terminate()方法生成的子进程来终止线程可能是最简单的...

class My_Thread(threading.Thread):

    def __init__(self):
        threading.Thread.__init__(self)
        self.process = None

    def run(self):
        print "Starting " + self.name
        cmd = [ "bash", 'process.sh']
        self.process = p = subprocess.Popen(cmd,
                     stdout=subprocess.PIPE,
                     stderr=subprocess.STDOUT)
        for line in iter(p.stdout.readline, b''):
            print ("-- " + line.rstrip())
        print "Exiting " + self.name

    def stop(self):
        print "Trying to stop thread "
        if self.process is not None:
            self.process.terminate()
            self.process = None

thr = My_Thread()
thr.start()
time.sleep(30)
thr.stop()
thr.join()

...导致SIGTERM被发送到bash,并且下一次调用p.stdout.readline()以引发异常,这将终止该帖子。

答案 1 :(得分:6)

Python线程不容易被杀死,你可以使用几乎相同的多处理模块(http://docs.python.org/2/library/multiprocessing.html)并且它具有终止()函数来杀死进程。

这是一个小例子,取自python docs。

>>> import multiprocessing, time, signal
>>> p = multiprocessing.Process(target=time.sleep, args=(1000,))
>>> print p, p.is_alive()
<Process(Process-1, initial)> False
>>> p.start()
>>> print p, p.is_alive()
<Process(Process-1, started)> True
>>> p.terminate()
>>> time.sleep(0.1)
>>> print p, p.is_alive()
<Process(Process-1, stopped[SIGTERM])> False
>>> p.exitcode == -signal.SIGTERM
True