108

我正在用 Python 编写一个命令行实用程序,因为它是生产代码,所以应该能够干净地关闭而不会将一堆东西(错误代码、堆栈跟踪等)转储到屏幕上。这意味着我需要捕捉键盘中断。

我尝试过使用 try catch 块,例如:

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

并捕捉信号本身(如本文所述):

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

这些错误不仅丑陋,而且帮助不大(尤其是对于没有源代码的最终用户)!

这个应用程序的清理代码相当大,所以这个问题很有可能会被真实用户遇到。有什么方法可以捕获或阻止此输出,还是我必须处理的事情?

4

2 回答 2

146

签出这个线程,它有一些关于退出和回溯的有用信息。

如果您对杀死程序更感兴趣,请尝试这样的事情(这也将消除清理代码下的腿):

if __name__ == '__main__':
    try:
        main()
    except KeyboardInterrupt:
        print('Interrupted')
        try:
            sys.exit(0)
        except SystemExit:
            os._exit(0)
于 2014-01-15T17:48:43.853 回答
9

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

于 2014-01-15T15:38:19.137 回答