我正在用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
这些错误不仅难看,而且对于没有源代码的最终用户来说也没有什么帮助!
该应用程序的清理代码相当庞大,因此存在实际用户遇到此问题的可能性。 是否有任何方法可以捕获或阻止此输出,还是我必须处理它?
sys.stdout
/sys.stderr
呢?比如sys.stderr = open(os.devnull, 'w')
。如果你真的不关心最终的输出,那这似乎是显而易见的解决方案。 - Bakuriuos._exit
但对我而言它看起来像是鼻妖。你的清理代码在哪里,你是否使用 atexit 模块来做清理工作? - wim/dev/null
。您可以编写一个自定义的文件对象,仅隐藏特定格式的消息。 - Bakuriu