在Python中,語法錯誤可以被Python解譯器發現,但邏輯上錯誤或變數使用錯誤卻不容易發現,如果結果沒有符合預期,則需要調試,一個很好的調試工具:Python內建的pdb模組。pdb是Python內建的調試模組。使用pdb模組可以為指令碼設定斷點、逐步執行、查看變數值等。
pdb可以用命令列參數的方式啟動,也可以使用import 將其匯入後再使用。
代碼如下:
>>> dir(pdb)
['Pdb', 'Repr', 'Restart', 'TESTCMD',.....,'re', 'run', 'runcall', 'runctx', 'runeval', 'set_trace', 'sys', 'test', 'traceback']
常見的pdb函數有以下幾個:
【pdb.run()函數】
>>> 該函數主要用於調試語句塊
>>> 基本用法如下
代碼如下:
>>> help(pdb.run)
Help on function run in module pdb:
run(statement, globals=None, locals=None)
>>>參數含義
statement: 要調試的語句塊,以字串的形式表示
globals: 選擇性參數,設定statement啟動並執行全域環境變數
locals: 選擇性參數,設定statement啟動並執行局部環境變數
>>>簡單樣本
代碼如下:
>>> import pdb # 匯入調試模組
>>> pdb.run(''''' # 調用run()函數執行一個for迴圈
for i in range(3):
i *= 3
print(i)
''')
> (2)()
(Pdb) n # (Pdb)為調試命令提示字元,表示可輸入調試命令
> (3)()
(Pdb) n # n(next)表示執行下一行
> (4)()
(Pdb) print(i) # 列印變數i的值
0
(Pdb) continue # 繼續運行程式
0
3
6
【pdb.runeval()函數】
>>>該函數主要用於調試運算式
>>>基本用法如下
代碼如下:
>>> help(pdb.runeval)
Help on function runeval in module pdb:
runeval(expression, globals=None, locals=None)
>>> 參數含義
expression: 要調試的,
globals: 選擇性參數,設定statement啟動並執行全域環境變數
locals: 選擇性參數,設定statement啟動並執行局部環境變數
>>> 簡單樣本
代碼如下:
>>> import pdb # 匯入pdb模組
>>> lst = [1, 2, 3] # 定義一個列表
>>> pdb.runeval('lst[1]') # 調用runaval()函數來調試運算式lst[1]
> (1)()
(Pdb) n # 進入調試狀態,使用n命令,逐步執行
--Return--
> (1)()->2
(Pdb) n # 逐步執行
2 # 返回運算式的值
>>> pdb.runeval('3 + 5*6/2') # 使用runaval()函數來調試運算式3+5*6/2
> (1)()->2
(Pdb) n
--Return--
> (1)()->18
(Pdb) n # 使用n命令逐步執行
18 # 最後得出運算式的值
【pdb.runcall()函數】
>>>該函數主要用於調試函數
>>>基本用法如下
代碼如下:
>>> help(pdb.runcall)
Help on function runcall in module pdb:
runcall(*args, **kwds)
>>> 參數含義
function: 函數名
args(kwds): 函數的參數
>>> 簡單樣本
代碼如下:
>>> import pdb # 匯入模組
>>> def sum(*args): # 定義函數sum,求所有參數之和
res = 0
for arg in args:
res += arg
return res
>>> pdb.runcall(sum, 1, 2, 3, 4) # 使用runcall調試函數sum
> (2)sum()
(Pdb) n # 進入調試狀態,逐步執行
> (3)sum()
(Pdb) n # 逐步執行
> (4)sum()
(Pdb) print(res) # 使用print列印res的值
0
(Pdb) continue # 繼續執行
10
>>> pdb.runcall(sum, 1, 2, 3, 4, 5, 6) # 調用runcall調試函數sum,參數不同
> (2)sum()
(Pdb) continue # 繼續執行
21 # 函數最後返回結果
【pdb.set_trace()函數】
>>>該函數主要用於指令碼中設定硬斷點
>>>基本用法如下
代碼如下:
>>> help(pdb.set_trace)
Help on function set_trace in module pdb:
set_trace()
>>>簡單樣本
代碼如下:
# file: test.py
import pdb
pdb.set_trace()
for i in range(5):
i *= 5
print(i)
運行指令碼後顯示:
代碼如下:
> d:\learn\python\test.py(6)()
-> for i in range(5):
(Pdb) list # 使用list列出指令碼內容
1 # file: test.py
2
3 import pdb
4
5 pdb.set_trace() # 使用set_trace()設定硬斷點
6 -> for i in range(5):
7 i *= 5
8 print(i)
[EOF] # 列出指令碼內容結束
(Pdb) continue # 使用continue繼續執行
0
5
10
15
20
【pdb調試命令】
pdb中的調試命令可以完成逐步執行、列印變數值、設定斷點等功能。pdb主要命令如下
代碼如下:
------------------------------------------------------------------------------
# 完整命令 簡寫命令 描述
------------------------------------------------------------------------------
# args a 列印當前函數的參數
# break b 設定斷點
# clear cl 清除斷點
# condition 無 設定條件斷點
# continue c 繼續運行,直到遇到斷點或者指令碼結束
# disable 無 禁用斷點
# enable 無 啟用斷點
# help h 查看pdb協助
# ignore 無 忽略斷點
# jump j 跳轉到指定行數運行
# list l 列出指令碼清單
# next n 執行下條語句,遇到函數不進入其內部
# print p 列印變數值
# quit q 退出pdb
# return r 一致運行到函數返回
# tbreak 無 設定臨時斷點、斷點只中斷一次
# step s 執行下一條語句,遇到函數進入其內部
# where w 查看所在的位置
# ! 無 在pdb中執行語句
>>>簡單樣本
代碼如下:
# -*- coding:gbk -*-
# file: prime.py
#
import math
# isprime函數判斷一個整數是否為素數
# 如果i能被2到i的平方根內的任意一個數整除,
# 則i不是素數,返回0,否則i是素數,返回1。
def isprime(i):
for t in range(2, int(math.sqrt(i)) + 1):
if i % t == 0:
return 0
print('100~110之間素數有: ')
for i in range(100, 110):
if isprime(i):
print(i)
先運行下面命令:
代碼如下:
d:\Learn\Python>python -m pdb prime.py
後輸入以下命令:
代碼如下:
d:\Learn\Python>python -m pdb prime.py
> d:\learn\python\prime.py(4)()
-> import math
(Pdb) list # 運行前面命令後停在這裡,list預設只列出11行
1 # -*- coding:gbk -*-
2 # file: prime.py
3 #
4 -> import math
5 # isprime函數判斷一個整數是否為素數
6 # 如果i能被2到i的平方根內的任意一個數整除,
7 # 則i不是素數,返回0,否則i是素數,返回1。
8 def isprime(i):
9 for t in range(2, int(math.sqrt(i)) + 1):
10 if i % t == 0:
11 return 0
(Pdb) l 14,17 # 使用list命令,列出14行,到17行
14 print('100~110之間素數有: ')
15 for i in range(100, 110):
16 if isprime(i):
17 print(i)
(Pdb) b 14 # 使用break命令設定斷點
Breakpoint 1 at d:\learn\python\prime.py:14 # 返回斷點編號: 1
(Pdb) b isprime # 在函數isprime設定斷點
Breakpoint 2 at d:\learn\python\prime.py:8 # 返回斷點編號: 2
(Pdb) c # 使用c命令運行運行指令碼
> d:\learn\python\prime.py(14)() # 停在斷點1處,即第14行
-> print('100~110之間素數有: ')
(Pdb) c # 使用c命令繼續運行指令碼
100~110之間素數有: # 第14行指令碼輸出
> d:\learn\python\prime.py(9)isprime() # 停在斷點2,即isprime函數處
-> for t in range(2, int(math.sqrt(i)) + 1):
(Pdb) b 15 # 在第15行處設定斷點
Breakpoint 3 at d:\learn\python\prime.py:15
(Pdb) disable 2 # 禁用斷點2,即isprime函數處的斷點
(Pdb) c # 使用c命令繼續運行指令碼
> d:\learn\python\prime.py(15)() # 停在斷點3處,即第15行
-> for i in range(100, 110):
(Pdb) print(i) # 使用print列印變數i的值
100
(Pdb) c # 繼續運行指令碼
> d:\learn\python\prime.py(15)()
-> for i in range(100, 110):
(Pdb) p i # 列印i的值
101
(Pdb) enable 2 # 恢複斷點2,即isprime函數處的斷點
(Pdb) c # 繼續運行指令碼
> d:\learn\python\prime.py(9)isprime()
-> for t in range(2, int(math.sqrt(i)) + 1):
(Pdb) n # 逐步執行下一條語句
> d:\learn\python\prime.py(10)isprime()
-> if i % t == 0:
(Pdb) print(t) # 使用print列印變數t的值
2
(Pdb) cl # 清楚所有斷點,輸入y確認
Clear all breaks? y
(Pdb) c # 繼續運行指令碼
103
105
107
109
(Pdb) q # 使用quit(q)退出pdb調試