python优雅编程_Python——traceback的优雅处理

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()

执行后输出如下:

1af16813c5322a6926c4e753bb9b0db5.png

通过示例,我们发现普通的打印异常只有很少量的信息(通常是异常的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()

执行后输出如下:

6a8ef342064490b6ffde2f114516a13c.png

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

输出:

ab0b916c82625d700655cc2d5cc2a135.png

这里我们可以发现打印的异常信息更加详细了,下面我们了解下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

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
东南亚位于我国倡导推进的“一带一路”海陆交汇地带,作为当今全球发展最为迅速的地区之一,近年来区域内生产总值实现了显著且稳定的增长。根据东盟主要经济体公布的最新数据,印度尼西亚2023年国内生产总值(GDP)增长5.05%;越南2023年经济增长5.05%;马来西亚2023年经济增速为3.7%;泰国2023年经济增长1.9%;新加坡2023年经济增长1.1%;柬埔寨2023年经济增速预计为5.6%。 东盟国家在“一带一路”沿线国家中的总体GDP经济规模、贸易总额与国外直接投资均为最大,因此有着举足轻重的地位和作用。当前,东盟与中国已互相成为双方最大的交易伙伴。中国-东盟贸易总额已从2013年的443亿元增长至 2023年合计超逾6.4万亿元,占中国外贸总值的15.4%。在过去20余年中,东盟国家不断在全球多变的格局里面临挑战并寻求机遇。2023东盟国家主要经济体受到国内消费、国外投资、货币政策、旅游业复苏、和大宗商品出口价企稳等方面的提振,经济显现出稳步增长态势和强韧性的潜能。 本调研报告旨在深度挖掘东南亚市场的增长潜力与发展机会,分析东南亚市场竞争态势、销售模式、客户偏好、整体市场营商环境,为国内企业出海开展业务提供客观参考意见。 本文核心内容: 市场空间:全球行业市场空间、东南亚市场发展空间。 竞争态势:全球份额,东南亚市场企业份额。 销售模式:东南亚市场销售模式、本地代理商 客户情况:东南亚本地客户及偏好分析 营商环境:东南亚营商环境分析 本文纳入的企业包括国外及印尼本土企业,以及相关上下游企业等,部分名单 QYResearch是全球知名的大型咨询公司,行业涵盖各高科技行业产业链细分市场,横跨如半导体产业链(半导体设备及零部件、半导体材料、集成电路、制造、封测、分立器件、传感器、光电器件)、光伏产业链(设备、硅料/硅片、电池片、组件、辅料支架、逆变器、电站终端)、新能源汽车产业链(动力电池及材料、电驱电控、汽车半导体/电子、整车、充电桩)、通信产业链(通信系统设备、终端设备、电子元器件、射频前端、光模块、4G/5G/6G、宽带、IoT、数字经济、AI)、先进材料产业链(金属材料、高分子材料、陶瓷材料、纳米材料等)、机械制造产业链(数控机床、工程机械、电气机械、3C自动化、工业机器人、激光、工控、无人机)、食品药品、医疗器械、农业等。邮箱:market@qyresearch.com

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值