Python——traceback的優雅處理


轉載聲明:本文轉載自https://www.cnblogs.com/sunsky303/p/9562269.html,版權歸原作者所有。

 

剛接觸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()

 

輸出:

 

看下定義:

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,因此只有一個層級被打印出來):

 

定義如下:

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

輸出:

 

 

從這個例子可以看出有時候我們想得到的是一個字符串,比如我們想通過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()

 

輸出如下:

 

這樣我們就得到了線程中的異常信息。

 

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM