为什么sys.exit()在Python中的线程内调用时不会退出?

时间:2009-05-25 03:10:45

标签: python python-2.6

这可能是一个愚蠢的问题,但我正在测试我对Python的一些假设,我很困惑为什么以下代码片段在线程中调用时不会退出,但是在调用时会退出主线。

import sys, time
from threading import Thread

def testexit():
    time.sleep(5)
    sys.exit()
    print "post thread exit"

t = Thread(target = testexit)
t.start()
t.join()
print "pre main exit, post thread exit"
sys.exit()
print "post main exit"

sys.exit()的文档说明调用应该从Python退出。我可以从这个程序的输出中看到“post thread exit”永远不会被打印出来,但是主线程在线程调用exit之后仍然继续运行。

是否为每个线程创建了一个单独的解释器实例,而对exit()的调用只是退出该单独的实例?如果是这样,线程实现如何管理对共享资源的访问?如果我确实想要从线程退出程序(不是我真的想要,但我只是理解的话)该怎么办?

6 个答案:

答案 0 :(得分:55)

sys.exit()引发SystemExit异常,thread.exit()也是如此。因此,当sys.exit()在该线程内引发该异常时,它与调用thread.exit()具有相同的效果,这就是为什么只有线程退出。

答案 1 :(得分:17)

  

如果我想要退出该程序该怎么办?   从线程?

除了Deestan所描述的方法,你可以调用os._exit(注意下划线)。在使用它之前,请确保您了解它 no 清理(例如调用__del__或类似)。

答案 2 :(得分:13)

  

如果我确实想要从线程中退出程序(不是我,那该怎么办?   实际上想要,但我理解这一点??

至少在Linux上你可以做类似的事情:

os.kill(os.getpid(), signal.SIGINT)

这将SIGINT发送到主线程,在那里它可以作为KeyboardInterrupt处理。

BTW:在Windows上,你只能发送一个SIGTERM信号,这个信号无法从Python中捕获。 (我更喜欢按照赫尔穆特的建议调用os._exit)

答案 3 :(得分:11)

“主要出口前,出线后退出”这个事实是打印出来困扰你的吗?

与其他语言(如Java)不同,模拟sys.exitSystem.exit,在Java的情况下)导致VM /进程/解释器立即停止,Python的sys.exit只是抛出例外:特别是SystemExit异常。

以下是sys.exit(仅print sys.exit.__doc__)的文档:

  

通过提高SystemExit(状态)退出解释器   如果省略状态或无,则默认为零(即成功)   如果状态为数字,则将其用作系统退出状态   如果是另一种物体,它将被打印并且系统为   退出状态将是一个(即失败)。

这有一些后果:

  • 在一个线程中它只会杀死当前线程,而不是整个进程(假设它一直到达堆栈顶部......)
  • 对象析构函数(__del__)可能会被引用,因为引用这些对象的堆栈帧已展开
  • 最后在堆栈展开时执行块
  • 您可以抓住SystemExit例外

最后一个可能是最令人惊讶的,这也是你在Python代码中几乎从不会有一个不合格的except语句的另一个原因。

答案 4 :(得分:11)

  

如果我确实要退出该计划该怎么办?   从线程(不是我实际上   我想,但我理解这一点??

我首选的方法是Erlang-ish消息传递。稍微澄清,我这样做:

import sys, time
import threading
import Queue # thread-safe

class CleanExit:
  pass

ipq = Queue.Queue()

def testexit(ipq):
  time.sleep(5)
  ipq.put(CleanExit)
  return

threading.Thread(target=testexit, args=(ipq,)).start()
while True:
  print "Working..."
  time.sleep(1)
  try:
    if ipq.get_nowait() == CleanExit:
      sys.exit()
  except Queue.Empty:
    pass

答案 5 :(得分:1)

_thread.interrupt_main() 自 Python 3.7 起可用(之前可选)

相关问题