Python——traceback的优雅处理

刚接触Python的时候,简单的异常处理已经可以帮助我们解决大多数问题;

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

#!/usr/bin/env python3

def func():

raise Exception("-- func exception --")

def main():

try:

func()

except Exception as e:

print(e)

if __name__ == "__main__":

main()

执行后输出如下:

通过示例,我们发现普通的打印异常只有很少量的信息(通常是异常的value值),这种情况下我们很难定位在哪块代码出的问题,以及如何出现这种异常。

那么到底要如何打印更加详细的信息呢?下面我们就来一一介绍。

sys.exc_info和traceback object

Python程序的traceback信息均来源于一个叫做 traceback object 的对象,而这个traceback

object通常是通过函数sys.exc_info()来获取的,先来看一个例子:

#!/usr/bin/env python3

import traceback

import sys

def func():

raise Exception("-- func exception --")

def main():

try:

func()

except Exception as e:

exc_type, exc_value, exc_obj = sys.exc_info()

print("exception_type: \t%s,\nexception_value: \t%s,\nexception_object: \t%s,\n" \

%(exc_type,exc_value,exc_obj))

if __name__ == "__main__":

main()

执行后输出如下:

通过以上示例我们可以看出,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

#!/usr/bin/env python3

import traceback

import sys

def func():

raise Exception("-- func exception --")

def main():

try:

func()

except Exception as e:

exc_type, exc_value, exc_obj = sys.exc_info()

traceback.print_tb(exc_obj)

if __name__ == "__main__":

main()

输出:

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

traceback.print_tb(tb[, limit[, file]])

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

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

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

#

print_exception

#!/usr/bin/env python3

import traceback

import sys

def func():

raise Exception("-- func exception --")

def main():

try:

func()

except Exception as e:

exc_type, exc_value, exc_obj = sys.exc_info()

traceback.print_exception(exc_type,exc_value,exc_obj,limit=2,file=sys.stdout)

if __name__ == "__main__":

main()

输出:

![](/media/images/blog/5fef9cb8f623dcd126ae837454cc734e.png)

看下定义:

traceback.print_exception(etype, value, tb[, limit[, file]])

跟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()来帮助获取这三个参数了,也因此这个函数是我们的程序中最常用的,因为它足够简单

#!/usr/bin/env python3

import traceback

import sys

def func():

raise Exception("-- func exception --")

def main():

try:

func()

except Exception as e:

exc_type, exc_value, exc_obj = sys.exc_info()

traceback.print_exc(limit=2,file=sys.stdout)

if __name__ == "__main__":

main()

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

![](/media/images/blog/74ed622f8c2f165044d87d7772520e4d.png)

定义如下:

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

只有两个参数,够简单

#

format_exc

#!/usr/bin/env python3

import traceback

import sys

import logging

logger = logging.getLogger("test_traceback")

def func():

raise Exception("-- func exception --")

def main():

try:

func()

except Exception as e:

exc_type, exc_value, exc_obj = sys.exc_info()

logger.error(traceback.format_exc(limit=1))

if __name__ == "__main__":

main()

输出:

![](/media/images/blog/9fb66b82e632449383692589143a944c.png)

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

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

#

获取线程中的异常信息

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

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

#!/usr/bin/env python3

import threading

import traceback

def func():

raise Exception("thread exception")

class ExceptionThread(threading.Thread):

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

"""

Redirect exceptions of thread to an exception handler.

"""

threading.Thread.__init__(self, group, target, name, args, kwargs, verbose)

if kwargs is None:

kwargs = {}

self._target = target

self._args = args

self._kwargs = kwargs

self._exc = None

def run(self):

try:

if self._target:

self._target()

except BaseException as e:

import sys

self._exc = sys.exc_info()

finally:

#Avoid a refcycle if the thread is running a function with

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

del self._target, self._args, self._kwargs

def join(self):

threading.Thread.join(self)

if self._exc:

msg = "Thread '%s' threw an exception: %s" % (self.getName(), self._exc[1])

new_exc = Exception(msg)

raise new_exc.__class__, new_exc, self._exc[2]

t = ExceptionThread(target=func, name='my_thread')

t.start()

try:

t.join()

except:

traceback.print_exc()

输出如下:

![](/media/images/blog/a8a39720571720dc96f730984084991c.png)

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

原文链接:https://www.cnblogs.com/oddcat/articles/11362961.html

python优雅编程_Python——traceback的优雅处理相关推荐

  1. python代码书写_Python代码的优雅写法,让代码更简洁

    我们都知道,Python 的设计哲学是「优雅」.「明确」.「简单」.这也许很多人选择 Python 的原因.但是我收到有些伙伴反馈,他写的 Python 并不优雅,甚至很臃肿,那可能是你的姿势不对哦! ...

  2. python socket编程_Python Socket编程实现网络编程

    对于有经验的开发人员来说,掌握的编程语言应该是不少的.在这些编程语言中,网络编程的应用时必不可少的.其中Python也是这样的编程语言.我们今天将会在这里为大家详细介绍一下Python Socket编 ...

  3. python gpu编程_Python笔记_第四篇_高阶编程_进程、线程、协程_5.GPU加速

    Numba:高性能计算的高生产率 在这篇文章中,笔者将向你介绍一个来自Anaconda的Python编译器Numba,它可以在CUDA-capable GPU或多核cpu上编译Python代码.Pyt ...

  4. python socket编程_Python学习记录-socket编程

    1. OSI七层模型详解 2. Python socket 什么是 Socket? Socket又称"套接字",应用程序通常通过"套接字"向网络发出请求或者应答 ...

  5. python树莓派编程_python树莓派编程

    广告关闭 腾讯云11.11云上盛惠 ,精选热门产品助力上云,云服务器首年88元起,买的越多返的越多,最高返5000元! 例如,你可以用树莓派搭建你自己的家用云存储服务器.? 树莓派用python来进行 ...

  6. python多线程编程_python多线程编程(1): python对多线程的支持

    前面介绍过多线程的基本概念,理解了这些基本概念,掌握python多线程编程就比较容易了. 在开始之前,首先要了解一下python对多线程的支持. 虚拟机层面 Python虚拟机使用GIL(Global ...

  7. python硬件编程_Python学习日记_《Python硬件编程实战》笔记_Mr_Ouyang

    书名: Python硬件编程实战 作者: 李茂 出版社: 机械工业出版社 [此处需要插入图片 Python封面] 笔者简评:不太适宜购买,全书大篇幅在用图片来解释极简单的细节,对于那些需要作者去深挖. ...

  8. python密码编程_Python密码学编程

    Python密码学编程 1 制作纸质加密工具 1.1 密码学是什么 1.2 代码与加密法 1.3 制作纸质加密轮盘 1.4 虚拟加密轮盘 1.5 如何使用加密轮盘加密 1.6 如何使用加密轮盘解密 1 ...

  9. python概率编程_Python概率编程库PyMC应用案例二则,pymc应用案例

    Python概率编程库PyMC应用案例二则,pymc应用案例 这是受国防科大刘万伟老师委托发的概率编程方面的内容,这方面我不懂,为了避免解释错了,我就直接把刘老师的PPT资料截图发了. 代码执行结果为 ...

最新文章

  1. 在Ubuntu 14.04 64bit上为Sublime text 3安装搜狗拼音输入法
  2. (转)CSS样式表继承详解
  3. 他从零开始学Python,25岁拥有480000粉丝:成功就靠这3点!
  4. 利用自定命令打开常用软件,小白秒变大神。
  5. java in action,java 7 in action
  6. .NET/C#使用NPOI操作Excel
  7. 新闻简报(7/18)
  8. 软件基本功:做自说明的测试文档,
  9. Echarts 下载使用教程
  10. 日系P2P原理探究(二) — Winny, 日本崛起
  11. 【主流Nivida显卡深度学习/强化学习/AI算力汇总】
  12. eclipse neno中tomcat配置servers locations地址的文件
  13. 台式关掉计算机不断网,笔记本电脑在关掉屏幕后不断网设置方法
  14. 不惧年龄,无限可能,32岁也能成功转行IT行业
  15. python生存曲线_生存曲线的估计方法(3):寿命表法
  16. PaddlePaddle - 人脸关键点检测课程笔记
  17. Win11怎么添加信任软件?Win11怎么把软件添加进白名单?
  18. Windows 找不到网络 计算机或设备,“win7系统宽带拨号提示找不到设备”的解决方案...
  19. 城市槽音乐在津巴布韦的美国音乐如何影响其他文化和身份的个案研究
  20. 小学信息技术用计算机画画说课,小学信息技术说课稿三篇

热门文章

  1. PL/SQL Developer远程连接Oracle数据库
  2. 日志分析工具Awstats实战之Apache篇-多站点日志分析
  3. .net 时间操作(datetime数据类型,datetime方法)
  4. html中位div添加水平线,html中div使用CSS实现水平/垂直居中的多种方式
  5. android通用adapter,Android通用ListViewAdapter的编写。
  6. java转动的风扇课程设计,课程设计—智能风扇设计报告
  7. 月份对比_行业洞察 | 10月份行业概览amp;头部广告主盘点
  8. python网盘系统_Python最新全套视频教程百度网盘资源
  9. matlab的单项悟空整流,一种新型单相脉冲整流器的MATLAB仿真研究
  10. 【面试】场景 智力题