使用Pdb調試Python

uwny2454 8年前發布 | 10K 次閱讀 Python Python開發

簡單介紹

Python自帶 Pdb庫,使用 Pdb調試 Python程序還是很方便的。但是遠程調試、多線程,Pdb是搞不定的

本文參考的相關文章如下:

用Pdb調試有多種方式

使用 Pdb調試 Python的程序的方式主要是下面的三種!下面逐一介紹

命令行加-m參數

命令行啟動目標程序,加上-m參數,這樣調用 testPdb.py的話斷點就是程序執行的第一行之前

本文接下來重點講到的實例展示就是使用這種方式進行調試的!

python -m pdb testPdb.py

在python交互環境調試

>>> import pdb
>>> import testPdb
>>> pdb.run('testPdb.test()')

代碼中插入一段程序

比較常用的,就是在程序中間插入一段程序,相對于在一般 IDE 里面打上斷點然后啟動 debug,不過這種方式是 hardcode的

if __name__ == "__main__":
  a = 1
  import pdb
  pdb.set_trace()
  b = 2
  c = a + b
  print(c)

然后正常運行腳本: python testPdb.py 到了 pdb.set_trace()那里就會定下來,然后就可以看到調試的提示符 (Pdb)了

針對上面的這段小程序的調試情況如下:

準備測試程序

接下來使用上面介紹的第一種方式來調試 Python程序,以此來介紹 pdb常用的命令,不過在開始之前先要準備好測試的程序代碼:

testFun.py

這是一個會被主模塊調用的子模塊,用于測試使用 Pdb調試的時候,是不是可以斷點從主模塊跟蹤進入子模塊(后續有說明)

#!/usr/bin/python

-- coding: utf-8 --

def add(a, b): return a + b</code></pre>

testPdb.py

這是下面被調試的主模塊的代碼

#!/usr/bin/python

-- coding: utf-8 --

def sub(a, b): return a - b

if name == "main":

print ''
import testFun
i = 0
a = 1
while(i < 100):
    a = testFun.add(a, 1)
    i = i + 1
print "累加結果:", a
print ""

for letter in 'Pdb':
    print "當前字母:", letter
print ""

fruits = ['banana', 'apple', 'mango']
for fruit in fruits:
    print "當前水果:", fruit
print ""


ret = 0
for num in range(10, 12):
    ret = sub(ret, num)
print '循環結果:', ret
print ""

d = {'abc': 123, 123: "abc"}
for (k,v) in d.items():
    print "當前鍵值對:", k, '-', v
print ""</code></pre> 

總結常用的命令

基礎命令

h(elp)命令:會打印當前版本 Pdb可用的命令,如果要查詢某個命令,可輸入 h [command] ,例如 h l 查看 list命令

l(ist)命令:可以列出當前將要運行的代碼塊

斷點管理

b(reak):設置斷點

比如 b 12 就是在當前腳本的第 9行加上斷點

比如 b sub 就是在當前腳本的 sub函數定義處加斷點

除了可以在當前的腳本中添加斷點之外,還可以在當前腳本對其他腳本下斷點,以上面用到的代碼為例 b testFun.add 就可以實現在 testFun.py腳本中的 add函數處加斷點

如果只用 b 就會顯示現有的全部斷點

condition bpnumber [condition]:設置條件斷點,比如 condition 2 a==0 ,就是在第二個斷點出加條件 “a==0”

cl(ear):刪除斷點,如果后面帶有參數,就是清楚指定的斷點;如果不帶參數就是清除所有的斷點

disable/enable:禁用/激活斷點

程序邏輯控制

下面展示的幾個命令,需要知道對應的腳本的代碼和行號,所以這里先截圖展示下面測試需要用到的前幾行代碼

c(ont(inue)),讓程序正常運行,直到遇到下一個斷點

n(ext),讓程序運行下一行,如果當前語句有一個函數調用,用n是不會進入被調用的函數體中的

下圖中展示的,當對腳本斷點調試到 testFun.add(a, 1)時,繼續執行n,并不會進入 testFun.add(a, 1)的函數內部

s(tep),跟n相似,但如果當前有一個函數調用,那么 s會進入被調用的函數體中

下圖中展示的,當對腳本斷點調試到 testFun.add(a, 1)時,繼續執行s,會進入 testFun.add(a, 1)對應的函數定義內部,雖然 testFun.add不是本腳本中定義的函數

j(ump),讓程序跳轉到指定的行數

假如當前所在行是 10,注意:假如執行了 j 20 之后,那么相當于程序直接跳到 20行,中間的 11~19行其實就直接跳過去根本沒有被執行到,所以如果這段代碼中有變量的聲明或對象的初始化需要在 20行及之后被用到,那么等到用到的時候就可能導致報錯!

打印重要信息

a(rgs),打印當前函數的參數。比如下圖就是展示斷點進入到 testFun.add內部之后,打印 testFun.add的參數

p,打印某個變量

退出調試

q,直接退出調試;或者使用 Ctrl+D的方式退出

最后說一句

上面展示的使用 Pdb調試的過程其實是很簡單的,文章中主要通過截圖展示運行的效果。如果單純的看一遍文章,不出意外,會很沒有頭緒,甚至感覺截圖中的命令、輸出亂七八糟,但是如果親自動手跟著走一遍流程,花不了一小時,但是效果絕對極佳!

 

來自:https://segmentfault.com/a/1190000006628456

 

 本文由用戶 uwny2454 自行上傳分享,僅供網友學習交流。所有權歸原作者,若您的權利被侵害,請聯系管理員。
 轉載本站原創文章,請注明出處,并保留原始鏈接、圖片水印。
 本站是一個以用戶分享為主的開源技術平臺,歡迎各類分享!