python出现traceback什么意思_Python Traceback详解

刚接触Python的时候,简单的异常处理已经可以帮助我们解决大多数问题,但是随着逐渐地深入,我们会发现有很多情况下简单的异常处理已经无法解决问题了,如下代码,单纯的打印异常所能提供的信息会非常有限。

1 deffunc1():2 raise Exception("--func1 exception--")3

4

5 defmain():6 try:7 func1()8 exceptException as e:9 printe10

11

12 if __name__ == '__main__':13 main()

View Code

执行后输出如下:

1 --func1 exception--

View Code

通过示例,我们发现普通的打印异常只有很少量的信息(通常是异常的value值),这种情况下我们很难定位在哪块代码出的问题,以及如何出现这种异常。那么到底要如何打印更加详细的信息呢?下面我们就来一一介绍。

sys.exc_info和traceback object

Python程序的traceback信息均来源于一个叫做traceback object的对象,而这个traceback object通常是通过函数sys.exc_info()来获取的,先来看一个例子:

importsysdeffunc1():raise NameError("--func1 exception--")defmain():try:

func1()exceptException as e:

exc_type, exc_value, exc_traceback_obj=sys.exc_info()print "exc_type: %s" %exc_typeprint "exc_value: %s" %exc_valueprint "exc_traceback_obj: %s" %exc_traceback_objif __name__ == '__main__':

main()

View Code

执行后输出如下:

exc_type: exc_value:--func1 exception--exc_traceback_obj:

View Code

通过以上示例我们可以看出,sys.exc_info()获取了当前处理的exception的相关信息,并返回一个元组,元组的第一个数据是异常的类型(示例是NameError类型),第二个返回值是异常的value值,第三个就是我们要的traceback object.

有了traceback object我们就可以通过traceback module来打印和格式化traceback的相关信息,下面我们就来看下traceback module的相关函数。

traceback module

Python的traceback module提供一整套接口用于提取,格式化和打印Python程序的stack traces信息,下面我们通过例子来详细了解下这些接口:

print_tb

1 importsys2 importtraceback3

4

5 deffunc1():6 raise NameError("--func1 exception--")7

8

9 defmain():10 try:11 func1()12 exceptException as e:13 exc_type, exc_value, exc_traceback_obj =sys.exc_info()14 traceback.print_tb(exc_traceback_obj)15

16

17 if __name__ == '__main__':18 main()

View Code

输出:

1 File "", line 11, inmain2 func1()3 File "", line 6, infunc14 raise NameError("--func1 exception--")

View Code

这里我们可以发现打印的异常信息更加详细了,下面我们了解下print_tb的详细信息:

tb: 这个就是traceback object, 是我们通过sys.exc_info获取到的

limit: 这个是限制stack trace层级的,如果不设或者为None,就会打印所有层级的stack trace

file: 这个是设置打印的输出流的,可以为文件,也可以是stdout之类的file-like object。如果不设或为None,则输出到sys.stderr。

print_exception

输出:

看下定义:

跟print_tb相比多了两个参数etype和value,分别是exception type和exception value,加上tb(traceback object),正好是sys.exc_info()返回的三个值

另外,与print_tb相比,打印信息多了开头的"Traceback (most...)"信息以及最后一行的异常类型和value信息

还有一个不同是当异常为SyntaxError时,会有"^"来指示语法错误的位置

print_exc

print_exc是简化版的print_exception, 由于exception type, value和traceback object都可以通过sys.exc_info()获取,因此print_exc()就自动执行exc_info()来帮助获取这三个参数了,也因此这个函数是我们的程序中最常用的,因为它足够简单

1 importsys2 importtraceback3

4

5 deffunc1():6 raise NameError("--func1 exception--")7

8 deffunc2():9 func1()10

11 defmain():12 try:13 func2()14 exceptException as e:15 traceback.print_exc(limit=1, file=sys.stdout)16

17

18 if __name__ == '__main__':19 main()

View Code

输出(由于limit=1,因此只有一个层级被打印出来):

1 Traceback (most recent call last):2 File "", line 13, inmain3 func2()4 NameError: --func1 exception--

View Code

定义如下:

1 traceback.print_exc([limit[, file]])

View Code

只有两个参数,够简单

format_exc

1 importlogging2 importsys3 importtraceback4 logger = logging.getLogger("traceback_test")5

6 deffunc1():7 raise NameError("--func1 exception--")8

9 deffunc2():10 func1()11

12 defmain():13 try:14 func2()15 exceptException as e:16 logger.error(traceback.format_exc(limit=1, file=sys.stdout))17

18

19 if __name__ == '__main__':20 main()

View Code

从这个例子可以看出有时候我们想得到的是一个字符串,比如我们想通过logger将异常记录在log里,这个时候就需要format_exc了,这个也是最常用的一个函数,它跟print_exc用法相同,只是不直接打印而是返回了字符串。

traceback module中还有一些其它的函数,但因为并不常用,就不在展开来讲,感兴趣的同学可以看下参考链接中的文档。

获取线程中的异常信息

通常情况下我们无法将多线程中的异常带回主线程,所以也就无法打印线程中的异常,而通过上边学到这些知识,我们可以对线程做如下修改,从而实现捕获线程异常的目的。

以下示例来自weidong的博客文章,稍有修改(见参考链接)

1 importthreading2 importtraceback3

4 defmy_func():5 raise BaseException("thread exception")6

7

8 classExceptionThread(threading.Thread):9

10 def __init__(self, group=None, target=None, name=None, args=(), kwargs=None, verbose=None):11 """

12 Redirect exceptions of thread to an exception handler.13 """

14 threading.Thread.__init__(self, group, target, name, args, kwargs, verbose)15 if kwargs isNone:16 kwargs ={}17 self._target =target18 self._args =args19 self._kwargs =kwargs20 self._exc =None21

22 defrun(self):23 try:24 ifself._target:25 self._target()26 exceptBaseException as e:27 importsys28 self._exc =sys.exc_info()29 finally:30 #Avoid a refcycle if the thread is running a function with

31 #an argument that has a member that points to the thread.

32 delself._target, self._args, self._kwargs33

34 defjoin(self):35 threading.Thread.join(self)36 ifself._exc:37 msg = "Thread '%s' threw an exception: %s" % (self.getName(), self._exc[1])38 new_exc =Exception(msg)39 raise new_exc.__class__, new_exc, self._exc[2]40

41

42 t = ExceptionThread(target=my_func, name='my_thread')43 t.start()44 try:45 t.join()46 except:47 traceback.print_exc()

View Code

输出如下:

1 Traceback (most recent call last):2 File "/data/code/testcode/thread_exc.py", line 43, in

3 t.join()4 File "/data/code/testcode/thread_exc.py", line 23, inrun5 self._target()6 File "/data/code/testcode/thread_exc.py", line 5, inmy_func7 raise BaseException("thread exception")8 Exception: Thread 'my_thread' threw an exception: thread exception

View Code

这样我们就得到了线程中的异常信息。

参考链接

  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值