在程序关闭期间捕获 Python 中的 KeyboardInterrupt

我正在用 Python 编写一个命令行实用程序,因为它是生产代码,所以应该能够干净利落地关闭,而不会向屏幕倾倒大量内容(错误代码、堆栈跟踪等)。这意味着我需要捕捉键盘中断。

I've tried using both a try catch block like:

if __name__ == '__main__':
try:
main()
except KeyboardInterrupt:
print 'Interrupted'
sys.exit(0)

and catching the signal itself (as in this post):

import signal
import sys


def sigint_handler(signal, frame):
print 'Interrupted'
sys.exit(0)
signal.signal(signal.SIGINT, sigint_handler)

这两种方法在正常操作时似乎都能很好地工作。但是,如果中断发生在应用程序末尾的清理代码期间,Python 似乎总是会在屏幕上打印一些内容。捕捉到中断给予

^CInterrupted
Exception KeyboardInterrupt in <bound method MyClass.__del__ of <path.to.MyClass object at 0x802852b90>> ignored

而处理信号给出了哪一个呢

^CInterrupted
Exception SystemExit: 0 in <Finalize object, dead> ignored

或者

^CInterrupted
Exception SystemExit: 0 in <bound method MyClass.__del__ of <path.to.MyClass object at 0x802854a90>> ignored

这些错误不仅难看,而且没有多大帮助(特别是对于没有源代码的最终用户) !

这个应用程序的清理代码相当大,所以真正的用户很有可能会碰到这个问题。有没有什么方法可以捕捉或阻止这个输出,或者只是我必须处理的东西?

257624 次浏览

在启动关闭之后,可以通过在开始清理代码之前调用 signal.signal(signal.SIGINT, signal.SIG_IGN)来忽略 SIGINT。

检查 this thread,它有一些关于退出和回溯的有用信息。

如果你只是对终止程序更感兴趣,试试下面的方法(这也会使清理代码中的代码失效) :

if __name__ == '__main__':
try:
main()
except KeyboardInterrupt:
print('Interrupted')
try:
sys.exit(0)
except SystemExit:
os._exit(0)