Python traceback的優雅處理

來源:互聯網
上載者:User

剛接觸Python的時候,簡單的異常處理已經可以協助我們解決大多數問題,但是隨著逐漸地深入,我們會發現有很多情況下簡單的異常處理已經無法解決問題了,如下代碼,單純的列印異常所能提供的資訊會非常有限。

def func1():    raise Exception("--func1 exception--")def main():    try:        func1()    except Exception as e:        print eif __name__ == '__main__':    main()

執行後輸出如下:

--func1 exception--

通過樣本,我們發現普通的列印異常只有很少量的資訊(通常是異常的value值),這種情況下我們很難定位在哪塊代碼出的問題,以及如何出現這種異常。那麼到底要如何列印更加詳細的資訊呢?下面我們就來一一介紹。

sys.exc_info和traceback object

Python程式的traceback資訊均來源於一個叫做traceback object的對象,而這個traceback object通常是通過函數sys.exc_info()來擷取的,先來看一個例子:

import sysdef 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_objif __name__ == '__main__':    main()

執行後輸出如下:

exc_type: <type 'exceptions.NameError'>exc_value: --func1 exception--exc_traceback_obj: <traceback object at 0x7faddf5d93b0>

通過以上樣本我們可以看出,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 sysimport tracebackdef 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 "<ipython-input-23-52bdf2c9489c>", line 11, in main    func1()File "<ipython-input-23-52bdf2c9489c>", 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 sysimport tracebackdef 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 "<ipython-input-24-a68061acf52f>", line 13, in main    func2()  File "<ipython-input-24-a68061acf52f>", 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 sysimport tracebackdef 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 "<ipython-input-25-a1f5c73b97c4>", line 13, in main    func2()NameError: --func1 exception--

定義如下:

traceback.print_exc([limit[, file]])
  • 只有兩個參數,夠簡單
format_exc
import loggingimport sysimport tracebacklogger = 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 threadingimport tracebackdef 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 <module>    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

這樣我們就得到了線程中的異常資訊。

參考連結

traceback官方文檔

weidong's blog

 

geekpy
連結:www.jianshu.com/p/a8cb5375171a
來源:簡書
簡書著作權歸作者所有,任何形式的都請聯絡作者獲得授權並註明出處。
相關文章

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.