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

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,816评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,729评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,300评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,780评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,890评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,084评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,151评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,912评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,355评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,666评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,809评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,504评论 4 334
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,150评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,882评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,121评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,628评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,724评论 2 351

推荐阅读更多精彩内容

  • 在C语言中,五种基本数据类型存储空间长度的排列顺序是: A)char B)char=int<=float C)ch...
    夏天再来阅读 3,339评论 0 2
  • 1. Python的hello-world: print ("Hello, Python!")、 完了 摇就完事儿...
    LunarShade阅读 1,379评论 0 0
  • 作者简介 高德拉特(Eliyahu M.Tolerate)、科克斯(JeffCOX) 高德拉特博士是以色列物理...
    壹路嚎叫再嚎叫阅读 1,038评论 0 1
  • 引言 罗辑思维在这周用了5期节目讲述《液晶演义》这个话题(未完)。这个话题之所以有料,一方面是因为液晶已经应用于生...
    海之方阅读 882评论 0 3
  • 很遗憾我没有那天写了这篇文章,拖到两天后的现在才写有一点点愧疚嘤嘤嘤。 早上还是一如往常起的很早,然后去吃了饭,结...
    酒与京城友阅读 115评论 0 0