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

Dan*_*Dan 57 python keyboardinterrupt

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

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

if __name__ == '__main__':
    try:
        main()
    except KeyboardInterrupt:
        print 'Interrupted'
        sys.exit(0)
Run Code Online (Sandbox Code Playgroud)

并捕捉信号本身(如在这篇文章中):

import signal
import sys

def sigint_handler(signal, frame):
    print 'Interrupted'
    sys.exit(0)
signal.signal(signal.SIGINT, sigint_handler)
Run Code Online (Sandbox Code Playgroud)

这两种方法在正常操作期间似乎都能很好地工作.但是,如果在应用程序结束时清理代码期间出现中断,Python似乎总是在屏幕上打印一些东西.捕获中断给出了

^CInterrupted
Exception KeyboardInterrupt in <bound method MyClass.__del__ of <path.to.MyClass object at 0x802852b90>> ignored
Run Code Online (Sandbox Code Playgroud)

而处理信号也给出了

^CInterrupted
Exception SystemExit: 0 in <Finalize object, dead> ignored
Run Code Online (Sandbox Code Playgroud)

要么

^CInterrupted
Exception SystemExit: 0 in <bound method MyClass.__del__ of <path.to.MyClass object at 0x802854a90>> ignored
Run Code Online (Sandbox Code Playgroud)

这些错误不仅难看,而且对于没有源代码的最终用户来说也不是很有帮助!

此应用程序的清理代码相当大,因此真正的用户可能会遇到此问题.有没有办法捕获或阻止此输出,或者它只是我必须处理的东西?

Dan*_*gan 87

签出这个帖子,它有一些关于退出和回溯的有用信息.

如果你对杀死程序更感兴趣,可以尝试这样的事情(这也将从清理代码中删除):

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

  • Linux 通常以 130: 128 + 2 http://tldp.org/LDP/abs/html/exitcodes.html#EXITCODESREF 退出,在 python 上找不到任何跨平台退出代码函数。但“1”绝对比“0”更好 (6认同)
  • 我建议除`as`外,然后获取并重用退出代码。 (3认同)
  • 是。绝对不要在KeyboardInterrupt上以0退出。如果有人在脚本,管道中使用您的脚本,或者他们会认为您的代码正常执行。 (2认同)
  • 如果用户从 bash 脚本调用 python 程序,并且他们在脚本中使用“set -e”(正如他们应该的那样),那么您希望在用户按 CTRL+C 后中断整个 bash 脚本。这需要返回非零退出代码。 (2认同)
  • @Bersan:谢谢!我在 https://www.grammarly.com/blog/than-then/ 上读过,现在我只需要记住它:D (2认同)
  • @Josiah - 这是一个超级老的线程(~8 年)。也就是说,当我在 2014 年写这篇文章时,我想到了一些特定的用例,我_认为_它与让 python 记录器完成刷新/关闭任何打开的文件等有关。请参考这两个调用之间的区别/sf/answers/671398171/ (2认同)

Dan*_*etz 6

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