這篇文章主要介紹“怎么使用pdb進行Python調(diào)試”,在日常操作中,相信很多人在怎么使用pdb進行Python調(diào)試問題上存在疑惑,小編查閱了各式資料,整理出簡單好用的操作方法,希望對大家解答”怎么使用pdb進行Python調(diào)試”的疑惑有所幫助!接下來,請跟著小編一起來學(xué)習(xí)吧!
成都創(chuàng)新互聯(lián)公司是一家專注于成都做網(wǎng)站、成都網(wǎng)站制作與策劃設(shè)計,開封網(wǎng)站建設(shè)哪家好?成都創(chuàng)新互聯(lián)公司做網(wǎng)站,專注于網(wǎng)站建設(shè)10余年,網(wǎng)設(shè)計領(lǐng)域的專業(yè)建站公司;建站業(yè)務(wù)涵蓋:開封等地區(qū)。開封做網(wǎng)站價格咨詢:18980820575
開始案例,我們首先探索pdb最簡單的使用:查閱一個變量的值。首先,我們在一個源碼文件中某一行,寫入下列語句:
import pdb; pdb.set_trace()
當(dāng)這一行代碼被執(zhí)行后,Python代碼文件就會被暫停運行,等待你發(fā)布命令來指導(dǎo)它下一步如何操作。運行上面的代碼后,你可以在命令行界面看到(Pdb)的提示符,這就意味著,代碼被停止了,等待命令的輸入。
自從Python3.7以來,官方標(biāo)準(zhǔn)建議使用標(biāo)準(zhǔn)庫內(nèi)部函數(shù)breakpoint()替代上面的代碼(注意:本文附帶的代碼都使用的上面代碼形式),這樣可以更加加快解釋器的運行和調(diào)試:
breakpoint()
默認情況下,breakpoint()將會倒入pdb模塊,然后調(diào)用pdb.set_trace()函數(shù),只是它進行了進一步封裝。但是,使用breakpoint()可以更加靈活,并且允許用戶通過調(diào)用它的API來控制調(diào)試行為,以及使用環(huán)境變量PYTHONBREAKPOINT。例如,當(dāng)我們設(shè)置PYTHONBREAKPOINT=0在我們的環(huán)境中,這就會完全關(guān)閉breakpoint()的功能,從而關(guān)閉調(diào)試功能。
此外,我們還可以不用手動在源代碼文件中加入斷點代碼,只需要在命令行輸入運行指令時通過參數(shù)傳遞來設(shè)置,例如:
$ python3 -m pdb app.py arg1 arg2
那么,讓我們直接進入本小節(jié)的內(nèi)容,也就是查閱代碼中變量的值,看下面的案例,使用的源碼文件是codeExample1.py:
#!/usr/bin/env python3 filename = __file__ import pdb; pdb.set_trace() print(f"path={filename}")
而在你命令行界面,運行上述的Python代碼,你就可以得到下面的輸出結(jié)果:
$ ./codeExample1.py > /code/codeExample1.py(5)() -> print(f"path={filename}") (Pdb)
接下來,讓我們輸入p filename這個命令,來查看filename這個變量的值,可以看到下述結(jié)果:
(Pdb) p filename "./codeExample1.py" (Pdb)
因為我們使用的是一個命令行接口界面程序(command-line interface),那么注意一下輸出的字符和格式,解釋如下:
>
開始的第一行告訴我們所運行的源碼文件名稱,源碼名稱之后,用小括號包含的是當(dāng)前代碼行數(shù),再后面就是函數(shù)名稱。這里,因為我們沒用調(diào)用任何函數(shù),而是處于模塊級別,所用見到的是
()。
->
開始的第二行表示目前行數(shù)代碼對應(yīng)的具體代碼內(nèi)容。
(Pdb)
是一個pdb提示符,等待下一個命令的輸入。
我們可以使用q命令,表示推出調(diào)試(quit)。
當(dāng)使用命令p,我們同樣可以輸入一個表達式,讓Python來計算表達式的值。 如果傳入一個變量名,pdb就會答應(yīng)當(dāng)前變量對應(yīng)的值。但是,我們可以進一步調(diào)查我們應(yīng)用程序當(dāng)前的運行狀態(tài)。
下面案例中,當(dāng)函數(shù)get_path()被調(diào)用,為了查看在這個函數(shù)中發(fā)生了什么,我已經(jīng)提前插入斷點程序pdb.set_trace(),來阻斷程序的運行,源碼codeExample2.py內(nèi)容如下:
#!/usr/bin/env python3 import os def get_path(filename): """Return file"s path or empty string if no path.""" head, tail = os.path.split(filename) import pdb; pdb.set_trace() return head filename = __file__ print(f"path = {get_path(filename)}")
如果在命令行中運行這個文件,可以得到下面的輸出:
$ ./codeExample2.py > /code/example2.py(10)get_path() -> return head (Pdb)
那么此時,我們處于什么階段:
>
:表示我們在源碼文件codeExample2.py的第10行代碼處,此處是函數(shù)get_path()。如果運行命令p輸出的標(biāo)量,就是當(dāng)前所引用的代碼幀,也就是當(dāng)前上下文內(nèi)的標(biāo)量。
->
:運行的代碼已經(jīng)被停止在return head處,這一行代碼還沒有被執(zhí)行,是位于codeExample2.py文件的第10行處,具體是在函數(shù)get_path()內(nèi)。
如果想要查看應(yīng)用當(dāng)前狀態(tài)代碼上下文情況,可以使用命令ll
(longlist),來查看代碼內(nèi)容,具體內(nèi)容如下:
(Pdb) ll 6 def get_path(filename): 7 """Return file"s path or empty string if no path.""" 8 head, tail = os.path.split(filename) 9 import pdb; pdb.set_trace() 10 -> return head (Pdb) p filename "./codeExample2.py" (Pdb) p head, tail (".", "codeExample2.py") (Pdb) p "filename: " + filename "filename: ./codeExample2.py" (Pdb) p get_path(Pdb) p getattr(get_path, "__doc__") "Return file"s path or empty string if no path." (Pdb) p [os.path.split(p)[1] for p in os.path.sys.path] ["pdb-basics", "python36.zip", "python3.6", "lib-dynload", "site-packages"] (Pdb)
你可以輸入任何有效的Python表達式接在p后面,來進行計算。當(dāng)你正在調(diào)試并希望在運行時直接在應(yīng)用程序中測試替代實現(xiàn)時,這尤其有用。您還可以使用命令 pp(漂亮打印)來美觀打印表達式。 如果您想打印具有大量輸出的變量或表達式,例如列表和字典。 如果可以,漂亮打印將對象保留在一行上,如果它們不適合允許的寬度,則將它們分成多行。
在這一部分,我們主要是用兩個命令來實現(xiàn)代碼的調(diào)試,如下圖所示:
命令n(next)和s(step)的區(qū)別是,pdb運行停止的位置。使用n(next),pdb會進行執(zhí)行,直到運行到當(dāng)前函數(shù)或模塊的下一行代碼,即:如果有外部函數(shù)被調(diào)用,不會跳轉(zhuǎn)到外部函數(shù)代碼中,可以把n理解為“step over”。使用s(step)來執(zhí)行當(dāng)前代碼,但是如果有外部函數(shù)被調(diào)用,會跳轉(zhuǎn)到外部函數(shù)中,可以理解為“step into”,如果執(zhí)行到外部跳轉(zhuǎn)函數(shù),s命令會輸出--Call--。
n(next)和s(step)命令都會暫定代碼的執(zhí)行,當(dāng)運行到當(dāng)前函數(shù)的結(jié)束部分,并且打印出--Return--,下面是codeExample3.py文件源碼內(nèi)容:
#!/usr/bin/env python3 import os def get_path(filename): """Return file"s path or empty string if no path.""" head, tail = os.path.split(filename) return head filename = __file__ import pdb; pdb.set_trace() filename_path = get_path(filename) print(f"path = {filename_path}")
如果在命令行中運行這個文件,同時輸入命令n,可以得到下面的輸出:
$ ./codeExample3.py > /code/example3.py(14)() -> filename_path = get_path(filename) (Pdb) n > /code/example3.py(15) () -> print(f"path = {filename_path}") (Pdb)
通過使用命令n(next),我們停止在15行代碼,并且我們也只是在此模塊中,沒有跳轉(zhuǎn)到函數(shù)get_path()。此處函數(shù)表示為(),表明我們目前是處于模塊級別而不是任何函數(shù)內(nèi)部。
再讓我們嘗試使用s(step)命令,輸出為:
$ ./codeExample3.py > /code/example3.py(14)() -> filename_path = get_path(filename) (Pdb) s --Call-- > /code/example3.py(6)get_path() -> def get_path(filename): (Pdb)
通過使用s(step)命令,我們停止在函數(shù)get_path()內(nèi)部的第6行代碼處,因為這個函數(shù)是在代碼文件中第14行處被調(diào)用的,注意在s命令之后輸出了--Call--,表明是函數(shù)調(diào)用。為了方便,pdb有命令記憶功能,如果我們要調(diào)試很多代碼,可以輸入Enter
回車鍵,來重復(fù)執(zhí)行命令。
下面是我們混合使用n(next)和s(step)命令的案例,首先輸入s(step),因為我們想要進入函數(shù)get_path(),然后通過命令n(next)在局部調(diào)試代碼,并且使用Enter
回車鍵,避免重復(fù)輸入命令:
$ ./codeExample3.py > /code/codeExample3.py(14)() -> filename_path = get_path(filename) (Pdb) s --Call-- > /code/codeExample3.py(6)get_path() -> def get_path(filename): (Pdb) n > /code/codeExample3.py(8)get_path() -> head, tail = os.path.split(filename) (Pdb) > /code/codeExample3.py(9)get_path() -> return head (Pdb) --Return-- > /code/codeExample3.py(9)get_path()->"." -> return head (Pdb) > /code/codeExample3.py(15) () -> print(f"path = {filename_path}") (Pdb) path = . --Return-- > /code/codeExample3.py(15) ()->None -> print(f"path = {filename_path}") (Pdb)
注意輸出的--Call--和--Return--,這是pdb輸出的信息,提示我們調(diào)試過程的狀態(tài)信息,n(next)和s(step)命令都會在函數(shù)返回后停止,這也就是我們會看到--Return--信息的輸出。此外,還要注意->"."
在第一個--Return--輸出上面的結(jié)尾處:
--Return-- > /code/example3.py(9)get_path()->"." -> return head (Pdb)
當(dāng)pdb停止到一個函數(shù)的結(jié)尾,但是還沒有運行到return時,pdb同樣會打印return值,在上面例子中就是"."
.
3.1顯示代碼
不要忘記,我們上面提到了命令ll
(longlist:顯示當(dāng)前函數(shù)或幀的源代碼),在我們調(diào)試進入到不熟悉的代碼上下文中,這個命令非常有效,我們可以打印出整個函數(shù)代碼,顯示樣例如下:
$ ./codeExample3.py > /code/codeExample3.py(14)() -> filename_path = get_path(filename) (Pdb) s --Call-- > /code/codeExample3.py(6)get_path() -> def get_path(filename): (Pdb) ll 6 -> def get_path(filename): 7 """Return file"s path or empty string if no path.""" 8 head, tail = os.path.split(filename) 9 return head (Pdb)
如果想要查看簡短的代碼片段,我們可以使用命令l
(list),不需要輸入?yún)?shù),它將會打印11行目前代碼附近的代碼內(nèi)容,案例如下:
$ ./codeExample3.py > /code/codeExample3.py(14)() -> filename_path = get_path(filename) (Pdb) l 9 return head 10 11 12 filename = __file__ 13 import pdb; pdb.set_trace() 14 -> filename_path = get_path(filename) 15 print(f"path = {filename_path}") [EOF] (Pdb) l [EOF] (Pdb) l . 9 return head 10 11 12 filename = __file__ 13 import pdb; pdb.set_trace() 14 -> filename_path = get_path(filename) 15 print(f"path = {filename_path}") [EOF] (Pdb)
斷點的正確使用,在我們調(diào)試過程中可以節(jié)約大量的時間。不需要單步調(diào)試一行行代碼,而只用簡單地在我們想要查閱的地方設(shè)置一個斷點,就可以直接運行到斷點處進行調(diào)試。同樣的,我們可以添加條件,來讓pdb判斷是否需要設(shè)置斷點。通常使用命令b(break)來設(shè)置一個斷點,我們可以通過指定代碼行數(shù),或者是想要調(diào)試的函數(shù)名稱,語法如下:
b(reak) [ ([filename:]lineno | function) [, condition] ]
如果filename:
沒有在代碼行數(shù)之前被指定,那么默認就是當(dāng)前代碼文件中。注意第二個可選參數(shù)是b: condition
,這個功能非常強大。假設(shè)在一個場景下,我們想要在某種條件成立的情況下設(shè)置斷點,如果我們傳入一個表達式座位第二個參數(shù),pdb會在計算改表達式為true的情況下設(shè)置斷點,我們會在下面給出案例。下面案例中,使用了一個工具模塊util.py,讓我們在函數(shù)get_path()中設(shè)置一個斷點,下面是代碼codeExample4.py的內(nèi)容:
#!/usr/bin/env python3 import util filename = __file__ import pdb; pdb.set_trace() filename_path = util.get_path(filename) print(f"path = {filename_path}")
下面是工具模塊util.py文件內(nèi)容:
def get_path(filename): """Return file"s path or empty string if no path.""" import os head, tail = os.path.split(filename) return head
首先,讓我們使用源碼文件名稱和代碼行數(shù)來設(shè)置斷點:
$ ./example4.py > /code/example4.py(7)() -> filename_path = util.get_path(filename) (Pdb) b util:5 Breakpoint 1 at /code/util.py:5 (Pdb) c > /code/util.py(5)get_path() -> return head (Pdb) p filename, head, tail ("./example4.py", ".", "example4.py") (Pdb)
命令c(continue)是實現(xiàn)被斷點停止后繼續(xù)運行的命令,下面,讓我們使用函數(shù)名來設(shè)置斷點:
$ ./codeExample4.py > /code/codeExample4.py(7)() -> filename_path = util.get_path(filename) (Pdb) b util.get_path Breakpoint 1 at /code/util.py:1 (Pdb) c > /code/util.py(3)get_path() -> import os (Pdb) p filename "./codeExample4.py" (Pdb)
如果輸入b,并且不帶任何參數(shù),就可以查看到所有已經(jīng)設(shè)置的斷點信息:
(Pdb) b Num Type Disp Enb Where 1 breakpoint keep yes at /code/util.py:1 (Pdb)
你可以使用命令 disable bpnumber 和 enable bpnumber 禁用和重新啟用斷點。 bpnumber 是斷點列表第一列 Num 中的斷點編號。 注意 Enb 列的值變化:
(Pdb) disable 1 Disabled breakpoint 1 at /code/util.py:1 (Pdb) b Num Type Disp Enb Where 1 breakpoint keep no at /code/util.py:1 (Pdb) enable 1 Enabled breakpoint 1 at /code/util.py:1 (Pdb) b Num Type Disp Enb Where 1 breakpoint keep yes at /code/util.py:1 (Pdb)
為了刪除一個斷點,可以使用命令cl(clear):
cl(ear) filename:lineno cl(ear) [bpnumber [bpnumber...]]
現(xiàn)在,讓我們嘗試在設(shè)置斷點時輸入表達式參數(shù),在當(dāng)前案例場景下,get_path()函數(shù)如果接受到一個相對路徑,即:如果路徑名稱不是以/開始的,那么就不會設(shè)置斷點,具體案例如下:
$ ./codeExample4.py > /code/codeExample4.py(7)() -> filename_path = util.get_path(filename) (Pdb) b util.get_path, not filename.startswith("/") Breakpoint 1 at /code/util.py:1 (Pdb) c > /code/util.py(3)get_path() -> import os (Pdb) a filename = "./codeExample4.py" (Pdb)
如果創(chuàng)建一個斷點后,繼續(xù)輸入c(continue)命令來運行,pdb只會在表達式計算為true的情況下停止運行。命令a(args)會打印出當(dāng)前函數(shù)的傳入?yún)?shù)。
上述案例中,如果你通過函數(shù)名稱而不是代碼行數(shù)來設(shè)置斷點,注意表達式只會使用當(dāng)前函數(shù)的參數(shù)或者全局變量,不然的話,斷點就會不去計算表達式,而直接停止函數(shù)運行。如果,我們還是想用不是當(dāng)前的函數(shù)變量來計算表達式,也就是使用的變量不在當(dāng)前函數(shù)參數(shù)列表中,那么就要指定代碼行數(shù),案例如下:
$ ./codeExample4.py > /code/codeExample4.py(7)() -> filename_path = util.get_path(filename) (Pdb) b util:5, not head.startswith("/") Breakpoint 1 at /code/util.py:5 (Pdb) c > /code/util.py(5)get_path() -> return head (Pdb) p head "." (Pdb) a filename = "./codeExample4.py" (Pdb)
5.繼續(xù)執(zhí)行代碼
目前,我們可以使用n(next)和s(step)命令來調(diào)試查看代碼,然后使用命令b(break)和c(continue)來停止或繼續(xù)代碼的運行,這里還有一個相關(guān)的命令:unt(until)。使用unt命令類似于c命令,但是它的運行結(jié)果是,下一行比當(dāng)前代碼行數(shù)大的位置。有時,unt更加方便和便捷使用,讓我們在下面案例中展示,首先給出使用語法:
取決于我們是否輸入代碼行數(shù)參數(shù)lineno,unt命令可以按照下面兩種方式運行:
沒有l(wèi)ineno,代碼可以繼續(xù)執(zhí)行到,下一次行數(shù)比當(dāng)前行數(shù)大的位置,這就相似于n(next),也就是類似于執(zhí)行“step over”的另一種形式。但是,命令n和unt的不同點是,unt只會在下一次行數(shù)比當(dāng)前行數(shù)大的位置停止,而n命令會停在下一個邏輯執(zhí)行行。
具有l(wèi)ineno,代碼會運行到下一次行數(shù)比當(dāng)前行數(shù)大或者相等的位置,這就類似于c(continue)后面帶有一個代碼函數(shù)參數(shù)。
上面兩種情況下,unt命令類似于n(next)和s(step)都只會停止在當(dāng)前幀(或函數(shù))。
當(dāng)你想繼續(xù)執(zhí)行并在當(dāng)前源文件中更遠的地方停止時,請使用 unt。 你可以將其視為 n (next) 和 b (break) 的混合體,具體取決于是否傳遞行號參數(shù)。
在下面的示例中,有一個帶有循環(huán)的函數(shù)。 在這里,我們希望繼續(xù)執(zhí)行代碼并在循環(huán)后停止,而不是單步執(zhí)行循環(huán)的每次迭代或設(shè)置斷點,下面是文件codeExample4unt.py文件的內(nèi)容:
#!/usr/bin/env python3 import os def get_path(fname): """Return file"s path or empty string if no path.""" import pdb; pdb.set_trace() head, tail = os.path.split(fname) for char in tail: pass # Check filename char return head filename = __file__ filename_path = get_path(filename) print(f"path = {filename_path}")
以及命令下使用unt輸出如下:
$ ./codeExample4unt.py > /code/codeExample4unt.py(9)get_path() -> head, tail = os.path.split(fname) (Pdb) ll 6 def get_path(fname): 7 """Return file"s path or empty string if no path.""" 8 import pdb; pdb.set_trace() 9 -> head, tail = os.path.split(fname) 10 for char in tail: 11 pass # Check filename char 12 return head (Pdb) unt > /code/codeExample4unt.py(10)get_path() -> for char in tail: (Pdb) > /code/codeExample4unt.py(11)get_path() -> pass # Check filename char (Pdb) > /code/codeExample4unt.py(12)get_path() -> return head (Pdb) p char, tail ("y", "codeExample4unt.py")
ll
命令首先用于打印函數(shù)的源代碼,然后是 unt。 pdb 記得上次輸入的命令,所以我只是按 Enter 重復(fù) unt 命令。 這將繼續(xù)執(zhí)行代碼,直到到達比當(dāng)前行大的源代碼行。
請注意,在上面的控制臺輸出中,pdb 僅在第 10 行和第 11 行停止了一次。由于使用了 unt,因此僅在循環(huán)的第一次迭代中停止執(zhí)行。 但是,循環(huán)的每次迭代都被執(zhí)行。 這可以在輸出的最后一行進行驗證。 char 變量的值 "y" 等于 tail 值 "codeExample4unt.py" 中的最后一個字符。
6.顯示表達式
類似于打印表達式p和pp的功能,我們可以使用dispaly [expression]來告訴pdb來顯示一個表達的值,同樣適用undisplay [expression]來清楚一個表達式顯示,下面使用一些使用語法解釋:
下面是一個案例,代碼文件為codeExample4display.py,展示了一個循環(huán)的使用:
$ ./codeExample4display.py > /code/codeExample4display.py(9)get_path() -> head, tail = os.path.split(fname) (Pdb) ll 6 def get_path(fname): 7 """Return file"s path or empty string if no path.""" 8 import pdb; pdb.set_trace() 9 -> head, tail = os.path.split(fname) 10 for char in tail: 11 pass # Check filename char 12 return head (Pdb) b 11 Breakpoint 1 at /code/codeExample4display.py:11 (Pdb) c > /code/codeExample4display.py(11)get_path() -> pass # Check filename char (Pdb) display char display char: "e" (Pdb) c > /code/codeExample4display.py(11)get_path() -> pass # Check filename char display char: "x" [old: "e"] (Pdb) > /code/codeExample4display.py(11)get_path() -> pass # Check filename char display char: "a" [old: "x"] (Pdb) > /code/codeExample4display.py(11)get_path() -> pass # Check filename char display char: "m" [old: "a"]
在上面的輸出中,pdb 自動顯示了 char 變量的值,因為每次遇到斷點時,它的值都會發(fā)生變化。 有時這很有幫助并且正是你想要的,但還有另一種使用顯示的方法。
你可以多次輸入 display 以構(gòu)建表達式監(jiān)視列表。 這比 p 更容易使用。 添加你感興趣的所有表達式后,只需輸入 display 即可查看當(dāng)前值:
$ ./codeExample4display.py > /code/codeExample4display.py(9)get_path() -> head, tail = os.path.split(fname) (Pdb) ll 6 def get_path(fname): 7 """Return file"s path or empty string if no path.""" 8 import pdb; pdb.set_trace() 9 -> head, tail = os.path.split(fname) 10 for char in tail: 11 pass # Check filename char 12 return head (Pdb) b 11 Breakpoint 1 at /code/codeExample4display.py:11 (Pdb) c > /code/codeExample4display.py(11)get_path() -> pass # Check filename char (Pdb) display char display char: "e" (Pdb) display fname display fname: "./codeExample4display.py" (Pdb) display head display head: "." (Pdb) display tail display tail: "codeExample4display.py" (Pdb) c > /code/codeExample4display.py(11)get_path() -> pass # Check filename char display char: "x" [old: "e"] (Pdb) display Currently displaying: char: "x" fname: "./codeExample4display.py" head: "." tail: "codeExample4display.py"
7.Python調(diào)用者ID
在最后一節(jié),我們會基于上述學(xué)習(xí)到的內(nèi)容,來演示“call ID”的功能使用。下面是案例代碼codeExample5.py的內(nèi)容:
#!/usr/bin/env python3 import fileutil def get_file_info(full_fname): file_path = fileutil.get_path(full_fname) return file_path filename = __file__ filename_path = get_file_info(filename) print(f"path = {filename_path}")
以及工具模塊fileutil.py文件內(nèi)容:
def get_path(fname): """Return file"s path or empty string if no path.""" import os import pdb; pdb.set_trace() head, tail = os.path.split(fname) return head
在這種情況下,假設(shè)有一個大型代碼庫,其中包含一個實用程序模塊 get_path() 中的函數(shù),該函數(shù)使用無效輸入進行調(diào)用。 但是,它是從不同包中的許多地方調(diào)用的。
如何查看調(diào)用程序是誰?
使用命令w(where)來打印一個代碼棧序列,就是從低到上顯示所有堆棧:
$ ./codeExample5.py > /code/fileutil.py(5)get_path() -> head, tail = os.path.split(fname) (Pdb) w /code/codeExample5.py(12)() -> filename_path = get_file_info(filename) /code/codeExample5.py(7)get_file_info() -> file_path = fileutil.get_path(full_fname) > /code/fileutil.py(5)get_path() -> head, tail = os.path.split(fname) (Pdb)
如果這看起來令人困惑,或者你不確定堆棧跟蹤或幀是什么,請不要擔(dān)心。 我將在下面解釋這些術(shù)語。 這并不像聽起來那么困難。
由于最近的幀在底部,從那里開始并從下往上閱讀。 查看以 -> 開頭的行,但跳過第一個實例,因為 pdb.set_trace() 用于在函數(shù) get_path() 中輸入 pdb。 在此示例中,調(diào)用函數(shù) get_path() 的源代碼行是:
-> file_path = fileutil.get_path(full_fname)
在每個 ->
上面的行包含文件名和代碼行數(shù)(在括號中),以及函數(shù)名稱,所以調(diào)用者就是:
/code/example5.py(7)get_file_info() -> file_path = fileutil.get_path(full_fname)
顯然在這個簡單的樣例中,我們展示了如何查找函數(shù)調(diào)用者,但是想象一個大型應(yīng)用程序,您在其中設(shè)置了一個帶有條件的斷點,以確定錯誤輸入值的來源,下面讓我們進一步深入。
什么是堆棧追蹤和棧幀內(nèi)容?
堆棧跟蹤只是 Python 為跟蹤函數(shù)調(diào)用而創(chuàng)建的所有幀的列表。 框架是 Python 在調(diào)用函數(shù)時創(chuàng)建并在函數(shù)返回時刪除的數(shù)據(jù)結(jié)構(gòu)。 堆棧只是在任何時間點的幀或函數(shù)調(diào)用的有序列表。 (函數(shù)調(diào)用)堆棧在應(yīng)用程序的整個生命周期中隨著函數(shù)被調(diào)用然后返回而增長和縮小。打印時,這個有序的幀列表,即堆棧,稱為堆棧跟蹤。 你可以隨時通過輸入命令 w 來查看它,就像我們在上面找到調(diào)用者一樣。
當(dāng)前棧幀什么意思?
將當(dāng)前幀視為 pdb 已停止執(zhí)行的當(dāng)前函數(shù)。 換句話說,當(dāng)前幀是你的應(yīng)用程序當(dāng)前暫停的位置,并用作 pdb 命令(如 p(打?。┑摹皡⒖紟?。p 和其他命令將在需要時使用當(dāng)前幀作為上下文。 在 p 的情況下,當(dāng)前幀將用于查找和打印變量引用。當(dāng) pdb 打印堆棧跟蹤時,箭頭 > 表示當(dāng)前幀。
怎么使用和切換棧幀?
你可以使用兩個命令 u(向上)和 d(向下)來更改當(dāng)前幀。 與 p 結(jié)合使用,這允許你在任何幀中調(diào)用堆棧的任何點檢查應(yīng)用程序中的變量和狀態(tài)。使用語法如下:
讓我們看一個使用 u 和 d 命令的例子。 在這種情況下,我們要檢查codeExample5.py 中函數(shù) get_file_info() 的局部變量 full_fname。 為此,我們必須使用命令 u 將當(dāng)前幀向上更改一級:
$ ./codeExample5.py > /code/fileutil.py(5)get_path() -> head, tail = os.path.split(fname) (Pdb) w /code/codeExample5.py(12)() -> filename_path = get_file_info(filename) /code/codeExample5.py(7)get_file_info() -> file_path = fileutil.get_path(full_fname) > /code/fileutil.py(5)get_path() -> head, tail = os.path.split(fname) (Pdb) u > /code/codeExample5.py(7)get_file_info() -> file_path = fileutil.get_path(full_fname) (Pdb) p full_fname "./codeExample5.py" (Pdb) d > /code/fileutil.py(5)get_path() -> head, tail = os.path.split(fname) (Pdb) p fname "./codeExample5.py" (Pdb)
因為是在fileutil.py文件函數(shù)get_path()中設(shè)置的斷點程序pdb.set_trace(),所以當(dāng)前幀被設(shè)置在此處,如下面所示:
> /code/fileutil.py(5)get_path()
為了訪問和打印在codeExample5.py文件函數(shù)get_file_info()里的變量full_fname,命令u實現(xiàn)移動到前一個棧幀:
(Pdb) u > /code/codeExample5.py(7)get_file_info() -> file_path = fileutil.get_path(full_fname)
請注意,在上面 u 的輸出中,pdb 在第一行的開頭打印了箭頭 >。 這是 pdb,讓你知道幀已更改,并且此源位置現(xiàn)在是當(dāng)前幀。 現(xiàn)在可以訪問變量 full_fname。 另外,重要的是要意識到第 2 行以 -> 開頭的源代碼行已被執(zhí)行。 由于幀被移到堆棧上,fileutil.get_path() 已被調(diào)用。 使用 u,我們將堆棧向上移動(從某種意義上說,回到過去)到調(diào)用 fileutil.get_path() 的函數(shù) codeExample5.get_file_info()。
繼續(xù)這個例子,在打印 full_fname 之后,使用 d 將當(dāng)前幀移動到其原始位置,并打印 get_path() 中的局部變量 fname。如果我們愿意,我們可以通過將 count 參數(shù)傳遞給 u 或 d 來一次移動多個幀。 例如,我們可以通過輸入 u 2 移動到 codeExample5.py 中的模塊級別:
$ ./codeExample5.py > /code/fileutil.py(5)get_path() -> head, tail = os.path.split(fname) (Pdb) u 2 > /code/codeExample5.py(12)() -> filename_path = get_file_info(filename) (Pdb) p filename "./codeExample5.py" (Pdb)
到此,關(guān)于“怎么使用pdb進行Python調(diào)試”的學(xué)習(xí)就結(jié)束了,希望能夠解決大家的疑惑。理論與實踐的搭配能更好的幫助大家學(xué)習(xí),快去試試吧!若想繼續(xù)學(xué)習(xí)更多相關(guān)知識,請繼續(xù)關(guān)注創(chuàng)新互聯(lián)網(wǎng)站,小編會繼續(xù)努力為大家?guī)砀鄬嵱玫奈恼拢?/p>
當(dāng)前名稱:怎么使用pdb進行Python調(diào)試
URL網(wǎng)址:http://weahome.cn/article/ggidso.html