python脚本调试-pdb

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)()

-> 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

(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)()

-> import math

(Pdb)

• j line_no:(jump)设置语句跳转

(Pdb) j 6                                    #跳转到第六行语句

> /newtest/mytest.py(6)()

-> if num>2:

(Pdb)

• 直接输入Enter,会重复执行上一条命令

(Pdb) cl

Clear all breaks? y

(Pdb)                            #回车

Clear all breaks? y

(Pdb)

• q(uit),退出debug

你可能感兴趣的:(python脚本调试-pdb)