我正在用Python编写命令行实用程序,因为它是生产代码,所以应该能够干净地关闭而不会将一堆东西(错误代码,堆栈跟踪等)转储到屏幕上。这意味着我需要捕获键盘中断。
我尝试过使用try catch块,如:
if __name__ == '__main__':
try:
main()
except KeyboardInterrupt:
print 'Interrupted'
sys.exit(0)
并捕捉信号本身(如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
这些错误不仅很难看,而且对于没有源代码的最终用户来说也不是很有帮助!
此应用程序的清理代码相当大,因此真正的用户可能会遇到此问题。有没有办法捕获或阻止这个输出,或者它只是我必须处理的东西?
答案 0 :(得分:77)
结帐this thread,它有一些关于退出和追溯的有用信息。
如果你对杀死程序更感兴趣,可以尝试这样的事情(这也将从清理代码中删除):
if __name__ == '__main__':
try:
main()
except KeyboardInterrupt:
print('Interrupted')
try:
sys.exit(0)
except SystemExit:
os._exit(0)
答案 1 :(得分:5)
在开始清理代码之前,可以通过调用signal.signal(signal.SIGINT, signal.SIG_IGN)
来关闭关闭后的SIGINT。