python traceback 丢失_Python traceback 异常处理
刚接触Python的时候,简单的异常处理已经可以帮助我们解决大多数问题,但是随着逐渐地深入,我们会发现有很多情况下简单的异常处理已经无法解决问题了,如下代码,单纯的打印异常所能提供的信息会非常有限。
def func1():
raise Exception("--func1 exception--")
def main():
try:
func1()
except Exception as e:
print e
if __name__ == '__main__':
main()
执行后输出如下:
--func1 exception--
通过示例,我们发现普通的打印异常只有很少量的信息(通常是异常的value值),这种情况下我们很难定位在哪块代码出的问题,以及如何出现这种异常。那么到底要如何打印更加详细的信息呢?下面我们就来一一介绍。
sys.exc_info和traceback object
Python程序的traceback信息均来源于一个叫做traceback object的对象,而这个traceback object通常是通过函数sys.exc_info()来获取的,先来看一个例子:
import sys
def func1():
raise NameError("--func1 exception--")
def main():
try:
func1()
except Exception as e:
exc_type, exc_value, exc_traceback_obj = sys.exc_info()
print "exc_type: %s" % exc_type
print "exc_value: %s" % exc_value
print "exc_traceback_obj: %s" % exc_traceback_obj
if __name__ == '__main__':
main()
执行后输出如下:
exc_type:
exc_value: --func1 exception--
exc_traceback_obj:
通过以上示例我们可以看出,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
import sys
import traceback
def func1():
raise NameError("--func1 exception--")
def main():
try:
func1()
except Exception as e:
exc_type, exc_value, exc_traceback_obj = sys.exc_info()
traceback.print_tb(exc_traceback_obj)
if __name__ == '__main__':
main()
输出:
File "", line 11, in main
func1()
File "", line 6, in func1
raise NameError("--func1 exception--")
这里我们可以发现打印的异常信息更加详细了,下面我们了解下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
import sys
import traceback
def func1():
raise NameError("--func1 exception--")
def func2():
func1()
def main():
try:
func2()
except Exception as e:
exc_type, exc_value, exc_traceback_obj = sys.exc_info()
traceback.print_exception(exc_type, exc_value, exc_traceback_obj, limit=2, file=sys.stdout)
if __name__ == '__main__':
main()
输出:
Traceback (most recent call last):
File "", line 13, in main
func2()
File "", line 9, in func2
func1()
NameError: --func1 exception--
看下定义:
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()来帮助获取这三个参数了,也因此这个函数是我们的程序中最常用的,因为它足够简单
import sys
import traceback
def func1():
raise NameError("--func1 exception--")
def func2():
func1()
def main():
try:
func2()
except Exception as e:
traceback.print_exc(limit=1, file=sys.stdout)
if __name__ == '__main__':
main()
输出(由于limit=1,因此只有一个层级被打印出来):
Traceback (most recent call last):
File "", line 13, in main
func2()
NameError: --func1 exception--
定义如下:
traceback.print_exc([limit[, file]])
只有两个参数,够简单
format_exc
import logging
import sys
import traceback
logger = logging.getLogger("traceback_test")
def func1():
raise NameError("--func1 exception--")
def func2():
func1()
def main():
try:
func2()
except Exception as e:
logger.error(traceback.format_exc(limit=1, file=sys.stdout))
if __name__ == '__main__':
main()
从这个例子可以看出有时候我们想得到的是一个字符串,比如我们想通过logger将异常记录在log里,这个时候就需要format_exc了,这个也是最常用的一个函数,它跟print_exc用法相同,只是不直接打印而是返回了字符串。
traceback module中还有一些其它的函数,但因为并不常用,就不在展开来讲,感兴趣的同学可以看下参考链接中的文档。
获取线程中的异常信息
通常情况下我们无法将多线程中的异常带回主线程,所以也就无法打印线程中的异常,而通过上边学到这些知识,我们可以对线程做如下修改,从而实现捕获线程异常的目的。
以下示例来自weidong的博客文章,稍有修改(见参考链接)
import threading
import traceback
def my_func():
raise BaseException("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=my_func, name='my_thread')
t.start()
try:
t.join()
except:
traceback.print_exc()
输出如下:
Traceback (most recent call last):
File "/data/code/testcode/thread_exc.py", line 43, in
t.join()
File "/data/code/testcode/thread_exc.py", line 23, in run
self._target()
File "/data/code/testcode/thread_exc.py", line 5, in my_func
raise BaseException("thread exception")
Exception: Thread 'my_thread' threw an exception: thread exception
这样我们就得到了线程中的异常信息。
python traceback 丢失_Python traceback 异常处理相关推荐
- python traceback 丢失_Python异常模块traceback用法实例分析
本文实例讲述了python异常模块traceback用法.分享给大家供大家参考,具体如下: traceback模块被用来跟踪异常返回信息. 如下例所示: import traceback try: r ...
- python优雅编程_Python——traceback的优雅处理
Python--traceback的优雅处理 刚接触Python的时候,简单的异常处理已经可以帮助我们解决大多数问题: 但是随着逐渐地深入,我们会发现有很多情况下简单的异常处理已经无法解决问题了,如下 ...
- python traceback安装_Python Traceback详解
刚接触Python的时候,简单的异常处理已经可以帮助我们解决大多数问题,但是随着逐渐地深入,我们会发现有很多情况下简单的异常处理已经无法解决问题了,如下代码,单纯的打印异常所能提供的信息会非常有限. ...
- python traceback对象_Python traceback【转】
1. Python中的异常栈跟踪 Python,在2.x中,异常对象可以是任何对象,异常对象和异常栈是分开的. python中用于处理异常栈的模块是traceback模块,它提供了print_exce ...
- python 异常分类_python的异常处理
1. 异常 2. 异常种类 3. 异常处理 1. 异常 异常就是程序运行中发生的错误信号,如程序产生问题,而没有处理它,程序运行就将停止,py中错误触发的异常如下 2.异常种类 一般异常 Attrib ...
- python request处理_python requests异常处理
requests的所有异常归类: IOError RequestException HTTPError(RequestException) UnrewindableBodyError(RequestE ...
- python deepcopy 丢失_python 引用、copy 与deepcopy
一.引用 a=[1,2,3] b=a 此为列表引用,a和b的内存地址是一样的,都是指向列表[1,2,3].其实是一个列表,两个名字.操作其中一个变量改变列表,另外一个变量显示的列表也同时改变. a=[ ...
- python deepcopy 丢失_Python muliple deepcopy行为
像为类定义复制协议这样简单的事情呢?在import copy class Graph(object): def __init__(self): self.data = [1,2,3] self.man ...
- python traceback模块_Python中的traceback模块
import traceback try: raise SyntaxError, "traceback test" except: traceback.print_exc() 将会 ...
最新文章
- [Contest20171109]函数(lipshitz)
- JAVA知识基础(六):抽象
- sox处理mp3_使用SoX将mp3文件拆分为TIME秒
- webpack 报错 No PostCSS Config found 解决方案。
- 软件系统换服务器地址,软件系统换服务器地址
- H5 video 播放器demo代码
- UE4开发六:人物动画
- Can‘t open /dev/sdb1 exclusively. Mounted filesystem?
- java 导出复杂格式的 Excel 留着自己备用
- Pathview包:整合表达谱数据可视化KEGG通路
- HTML简单的个人博客网站 DIV学生网页设计作品 dreamweaver作业静态HTML网页设计模板 个人网页作业制作
- Linux系统默认PATH环境变量的设置
- python 3.8.5 打包_python做个word转pdf的小软件
- 夜雨寄北 文/江湖一劍客
- java环境jdk配置
- 可以参悟人生的十个故事
- Autodesk 开源 3D 打印机
- 《Java解惑》系列——02字符谜题——谜题17:嗯??
- 微信小程序request请求后台接收不到参数问题
- 药品药店管理系统c语言设计,java毕业设计_springboot框架的药店药品货架管理系统...
热门文章
- Dubbo -- Dubbo快速入门代码编写
- [项目1] bloger - day1
- c++ 保留有效数字和小数位
- python接口学习记录
- 超算中心和根服务器,华数将和下一代互联网国家工程中心共建IPv6根服务器
- 什么是Linux BSP?
- Cesium加载.shp三维城市建筑模型,并根据3DTiles建筑物高度修改颜色
- localdatetime 后一天_Java LocalDateTime获取前一天,—-JDK8新时间类的简单使用
- (九)串行口方式0 拓展并行输出端口 02 74LS164芯片
- 为西安地铁“让路” 47条公交线路有调整