如何通过日志模块而不是stderr导致未捕获的异常输出?

我意识到最好的办法是:

try:
    raise Exception, 'Throwing a boring exception'
except Exception, e:
    logging.exception(e)

但我的情况是,如果在没有捕获异常时自动调用logging.exception(…),那就太好了。


当前回答

下面是一个完整的小例子,还包括一些其他技巧:

import sys
import logging
logger = logging.getLogger(__name__)
handler = logging.StreamHandler(stream=sys.stdout)
logger.addHandler(handler)

def handle_exception(exc_type, exc_value, exc_traceback):
    if issubclass(exc_type, KeyboardInterrupt):
        sys.__excepthook__(exc_type, exc_value, exc_traceback)
        return

    logger.error("Uncaught exception", exc_info=(exc_type, exc_value, exc_traceback))

sys.excepthook = handle_exception

if __name__ == "__main__":
    raise RuntimeError("Test unhandled")

忽略KeyboardInterrupt,这样控制台python程序就可以用Ctrl + C退出。 完全依赖python的日志模块来格式化异常。 使用带有示例处理程序的自定义日志记录器。这一个将未处理的异常更改为stdout而不是stderr,但是您可以以相同的样式向记录器对象添加各种处理程序。

其他回答

为什么不:

import sys
import logging
import traceback

def log_except_hook(*exc_info):
    text = "".join(traceback.format_exception(*exc_info()))
    logging.error("Unhandled exception: %s", text)

sys.excepthook = log_except_hook

None()

下面是使用sys的输出。如上图所示:

$ python tb.py
ERROR:root:Unhandled exception: Traceback (most recent call last):
  File "tb.py", line 11, in <module>
    None()
TypeError: 'NoneType' object is not callable

下面是使用sys. js的输出。但是thook评论道:

$ python tb.py
Traceback (most recent call last):
  File "tb.py", line 11, in <module>
    None()
TypeError: 'NoneType' object is not callable

唯一的区别是前者在第一行的开头有ERROR:root:Unhandled异常。

为了回答在已接受答案的评论部分讨论的zeus先生的问题,我使用它来记录交互式控制台中未捕获的异常(使用PyCharm 2018-2019测试)。我发现了。excepthook不能在python shell中工作,所以我深入研究了一下,发现我可以使用sys。exc_info代替。然而,系统。与sys不同,Exc_info不带参数。除了thook,它有3个参数。

在这里,我使用了两个sys。除了thook和sys。Exc_info来记录交互式控制台和带有包装器函数的脚本中的两个异常。要将钩子函数附加到两个函数,我有两个不同的接口,这取决于是否给出参数。

代码如下:

def log_exception(exctype, value, traceback):
    logger.error("Uncaught exception occurred!",
                 exc_info=(exctype, value, traceback))


def attach_hook(hook_func, run_func):
    def inner(*args, **kwargs):
        if not (args or kwargs):
            # This condition is for sys.exc_info
            local_args = run_func()
            hook_func(*local_args)
        else:
            # This condition is for sys.excepthook
            hook_func(*args, **kwargs)
        return run_func(*args, **kwargs)
    return inner


sys.exc_info = attach_hook(log_exception, sys.exc_info)
sys.excepthook = attach_hook(log_exception, sys.excepthook)

日志记录设置可以在gnu_lorien的回答中找到。

以Jacinda的回答为基础,但使用记录器对象:

def catchException(logger, typ, value, traceback):
    logger.critical("My Error Information")
    logger.critical("Type: %s" % typ)
    logger.critical("Value: %s" % value)
    logger.critical("Traceback: %s" % traceback)

# Use a partially applied function
func = lambda typ, value, traceback: catchException(logger, typ, value, traceback)
sys.excepthook = func

包装你的应用程序入口调用在一个尝试…Except块,这样您就能够捕获和记录(可能还会重新引发)所有未捕获的异常。例如:

if __name__ == '__main__':
    main()

这样做:

if __name__ == '__main__':
    try:
        main()
    except Exception as e:
        logger.exception(e)
        raise

正如奈德指出的,系统。Excepthook在每次引发并未捕获异常时调用。这样做的实际含义是,您可以在代码中重写sys的默认行为。Excepthook可以做任何你想做的事情(包括使用logging.exception)。

举个稻草人的例子:

import sys
def foo(exctype, value, tb):
    print('My Error Information')
    print('Type:', exctype)
    print('Value:', value)
    print('Traceback:', tb)

覆盖sys.excepthook:

>>> sys.excepthook = foo

提交明显的语法错误(省略冒号),并返回自定义错误信息:

>>> def bar(a, b)
My Error Information
Type: <type 'exceptions.SyntaxError'>
Value: invalid syntax (<stdin>, line 1)
Traceback: None

有关sys. exe的详细信息。但是,看看这些文件。