python自带的pdb库,可以实现简单的调试功能,基本命令与gdb类似,pdb主要支持多断点设置(可条件设置),代码级单步调试,查看堆栈信息,代码查看。
Pdb的使用主要有以下几种
import pdb
直接在代码里需要开始调试的地方写入一个pdb.set_trace()语句,这样就可以设置一个断点,程序会在pdb.set_trace()处暂停并进入pdb调试环境(相当于IDE环境下设置断点),如下面一个判断某数字是否是素数的函数
#mytest.py文件
import pdb #导入pdb
import math
def func(num):
if num<=1:
return -1
pdb.set_trace() #设置断点
if num>2:
for i in range(2,math.ceil(math.sqrt(num))):
if num%i==0:
print ('是合数')
return
print ('是质数')
func(11)
运行该文件,如
[root@localhost newtest]# python3 mytest.py
> /newtest/mytest.py(8)func()
-> if num>2:
(Pdb)
可以看出,遇到断点后,程序进入调试模式,且执行流停留在接下来要执行的语句上。>后面信息指明了当前程序接下来将要运行哪一行;->后面的语句即是将要运行的语句;(Pdb)后面就可以输入调试命令了。
现在就可以用pdb 命令调试了,一些常用指令:
• h(elp) [comman] #打印可用指令及帮助信息
• n(ext) ,执行下一个语句,如果本句是函数,则把该函数当做一条语句,执行该函数然后返回
(Pdb) n
> /newtest/mytest.py(9)func()
-> for i in range(2,math.ceil(math.sqrt(num))): #尚未开始迭代
(Pdb) n
> /newtest/mytest.py(10)func()
-> if num%i==0:
(Pdb)
• s(tep) ,执行下一个语句,若为函数则进入函数体,指向函数体的第一句
• a(rgs) ,打印当前函数的参数
(Pdb) a
num = 11
(Pdb)
• p(rint) ,打印某个变量
(Pdb) p i
2
(Pdb) n
> /newtest/mytest.py(9)func()
-> for i in range(2,math.ceil(math.sqrt(num))): #11%2条件为假,又回到 for语句进行循环
(Pdb) n
> /newtest/mytest.py(10)func()
-> if num%i==0:
(Pdb) p i
3
(Pdb)
• unt(il),执行到下一行
作用是跳出循环,或者当前堆栈结束,比如遇到了一个for循环,我们想迅速运行完这个循环,就可以使用此命令
(Pdb) p i
2
(Pdb) unt #结束for循环
> /newtest/mytest.py(11)func()
-> print ('是质数')
(Pdb) p i
3
(Pdb)
可以发现执行until把整个循环走了一遍,然后才到下一行。
如果输入PDB不认识的命令,PDB会把他当做Python语句在当前环境下执行。这样我们可以人为修改某些变量,从而改变程序的行为,如
(Pdb) a
num = 11
(Pdb) num=21 #修改参数
(Pdb) s
> /newtest/mytest.py(4)func()
-> if num<=1:
(Pdb) s
> /newtest/mytest.py(6)func()
-> if num>2:
(Pdb) s
> /newtest/mytest.py(7)func()
-> for i in range(2,math.ceil(math.sqrt(num))):
(Pdb) s #进入循环
> /newtest/mytest.py(8)func()
-> if num%i==0:
(Pdb) until #循环运行结束
> /newtest/mytest.py(9)func()
-> print ('是合数')
(Pdb) s
是合数
> /newtest/mytest.py(10)func()
-> return
(Pdb)
-m pdb
上面的方式,需要改动源代码文件,多有不便。我们可以通过命令 python -m pdb xxx.py 启动脚本,进入单步执行模式。如
[root@localhost newtest]# python3 -m pdb mytest.py
> /newtest/mytest.py(2)<module>()
-> import math #将要执行脚本第一条语句
(Pdb)
这种调试模式下,一些常用指令:
• b(reak),添加断点
b line_no:当前脚本的line_no行添加断点
(Pdb) b 6 # if num>2: 处设置断点
Breakpoint 1 at /newtest/mytest.py:6
(Pdb)
b 列出当前所有断点,和断点执行到统计次数
(Pdb) b
Num Type Disp Enb Where
1 breakpoint keep yes at /newtest/mytest.py:6
(Pdb)
其中,num显示了断点编号。
b filename:line_no:脚本filename的line_no行添加断点
b function:在函数function的第一条可执行语句处添加断点
• cl(ear),清除断点
cl 清除所有断点
cl bpnumber1 bpnumber2... 清除断点号为bpnumber1,bpnumber2...的断点
(Pdb) cl 1 #删除第一个断点
Deleted breakpoint 1 at /newtest/mytest.py:6
(Pdb)
• c(ontinue)继续执行,直到遇到下一条断点
• l(ist) ,查看指定代码段
first_line_no, last_line_no,列出first--second范围的代码,如果second<first,second将被解析为行数
(Pdb) list 6,8
6 B if num>2:
7 for i in range(2,math.ceil(math.sqrt(num))):
8 if num%i==0:
(Pdb)
如果没有指定范围,则列出当前执行语句周围11条代码。
• run:重新启动debug,相当于restart
(Pdb) run
Restarting mytest.py with arguments:
mytest.py
> /newtest/mytest.py(2)<module>()
-> import math
(Pdb)
• j line_no:(jump)设置语句跳转
(Pdb) j 6 #跳转到第六行语句
> /newtest/mytest.py(6)<module>()
-> if num>2:
(Pdb)
• 直接输入Enter,会重复执行上一条命令
(Pdb) cl
Clear all breaks? y
(Pdb) #回车
Clear all breaks? y
(Pdb)
• q(uit),退出debug