Python 程序報錯崩潰後,如何倒回到崩潰的位置?

假設我們有一段程序,從 Redis 中讀取數據,解析以後提取出裏面的 name 字段:


import json
import redis


client = redis.Redis()
def read():
    while True:
        data = client.lpop('info')
        if data:
            yield json.loads(data)
        else:
            break


def parse():
    for data in self.read():
        print(data['name'])


if __name__ == '__main__':
    parse()

代碼的邏輯本身很簡單,從 Redis 中一條一條讀取數據,讀到的數據是 JSON 字符串,所以先使用json.loads解析成字典。然後讀取字典中的name對應的值。一直讀到Redis 列表爲空。

我們運行一下看看:

報錯了,說明Redis 中的某一條數據有問題。你想看看這條有問題的數據,但是現在程序已經崩潰了,進程結束了,這條有問題的數據也就永久丟失了。你再也不可能知道它長什麼樣了。

玩過《火焰紋章-風花雪月》的朋友都知道,主角有一個技能叫做天刻之脈動,如果隊友死了,他可以逆轉時間,回到隊友被殺之前,從而改變隊友的命運。

那麼,在Python裏面我們有沒有什麼辦法讓程序起死回生,看到當初導致程序報錯的那一行代碼呢?如果你是使用python3 xxx.py運行的程序,那麼確實,除非你能重新導入剛纔的數據,否則無法知道。

但是,如果你是使用如下命令:python3 -i xxx.py啓動的程序,那麼世界就不一樣了,你的程序獲得了起死回生的能力。你可以重新回到事故現場。

我們恢復一下 Redis 的數據(當然,在生產環境裏面你可能就沒有辦法恢復了。但現在寫文章的示例數據,我還是可以回覆的_

然後使用python3 -i read_name.py重新運行這個程序:

可以看到,現在雖然程序崩潰了,但是卻出現了 Python 的交互環境。進程並沒有完全退出。這樣一來,我們就可以輸入魔法指令,讓程序倒退回到報錯的那個地方。輸入命令:

import pdbpdb.pm()

運行效果如下圖所示:

現在,我們已經回到了報錯的那一行了。報錯報的是 data這個字典沒有name這個 key,那麼我們就來看看這個字典裏面有什麼,直接輸入變量名 data:

原來,這一條有問題的數據,是把name寫成了name1。

總結

pdb是Python自帶的調試工具。我們使用的PyCharm的調試功能,也是基於pdb實現的。

爲解決初學者學習上的困難,專門建立的Python學習扣qun:784758214,從0基礎的python腳本到web開發、爬蟲、django、數據挖掘數據分析等,0基礎到項目實戰的資料都有整理。送給每一位python的小夥伴!每晚分享一些學習的方法和需要注意的小細節,學習路線規劃,利用編程賺外快。點擊加入我們的 python學習圈

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章