使用pdb模块调试Python

在Python中,我们需要debug时,有三种方式:

  • 加log语句。最简单的方式是添加print()语句来输出我们想要获知的状态或者变量,好处是简单容易操作,坏处是debug完了之后,还需要将print语句注释掉;进阶的方法是使用logging模块写log语句,这样做的好处是:能把log语句分级,并且能够根据参数决定输出什么样层级的log,用这种方法可以让我们不用注释掉log语句。
  • 使用IDE(Intergrated Development Environment, 集成开发环境),比如Pycharm,这类的IDE可以像visual studio一样帮助我们debug。
  • 使用pdb这个python自带的模块来帮助我们debug。

使用 pdb 进行调试

pdb 是 python 自带的一个包,为 python 程序提供了一种交互的源代码调试功能,主要特性包括设置断点、单步调试、进入函数调试、查看当前代码、查看栈片段、动态改变变量的值等。pdb 提供了一些常用的调试命令,详情见表 1。

表 1. pdb 常用命令
命令 解释
break 或 b 设置断点 设置断点
continue 或 c 继续执行程序
list 或 l 查看当前行的代码段
step 或 s 进入函数
return 或 r 执行代码直到从当前函数返回
exit 或 q 中止并退出
next 或 n 执行下一行
p <arg_name> 打印变量的值
help 帮助

下面结合具体的实例讲述如何使用 pdb 进行调试。

import pdb
a = 1
pdb.set_trace()
b = 2
a = a + bprint(a)

这段是最简单的pdb模块调试示例代码。pdb.set_trace()这里就是一个断点,程序运行之后,会停留在pdb.set_trace()这里。此时输入n再按回车键,可以执行下一行语句。后面直接按回车键,可继续执行一行。其他参数作用也是如此,以此类推。

清单 2. 利用 pdb 调试

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

[[email protected] ~]#  python epdb1.py

 > /root/epdb1.py(4)?()

 -> b = "bbb"

 (Pdb) n

 > /root/epdb1.py(5)?()

 -> c = "ccc"

 (Pdb)

 > /root/epdb1.py(6)?()

 -> final = a + b + c

 (Pdb) list

  1     import pdb

  2     a = "aaa"

  3     pdb.set_trace()

  4     b = "bbb"

  5     c = "ccc"

  6  -> final = a + b + c

  7     print final

 [EOF]

 (Pdb)

 [EOF]

 (Pdb) n

 > /root/epdb1.py(7)?()

 -> print final

 (Pdb)

退出 debug:使用 quit 或者 q 可以退出当前的 debug,但是 quit 会以一种非常粗鲁的方式退出程序,其结果是直接 crash。

清单 3. 退出 debug

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

[[email protected] ~]#  python epdb1.py

 > /root/epdb1.py(4)?()

 -> b = "bbb"

 (Pdb) n

 > /root/epdb1.py(5)?()

 -> c = "ccc"

 (Pdb) q

 Traceback (most recent call last):

  File "epdb1.py", line 5, in ?

    c = "ccc"

  File "epdb1.py", line 5, in ?

    c = "ccc"

  File "/usr/lib64/python2.4/bdb.py", line 48, in trace_dispatch

    return self.dispatch_line(frame)

  File "/usr/lib64/python2.4/bdb.py", line 67, in dispatch_line

    if self.quitting: raise BdbQuit

 bdb.BdbQuit

打印变量的值:如果需要在调试过程中打印变量的值,可以直接使用 p 加上变量名,但是需要注意的是打印仅仅在当前的 statement 已经被执行了之后才能看到具体的值,否则会报 NameError: < exceptions.NameError … ....> 错误。

清单 4. debug 过程中打印变量

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

[[email protected] ~]#  python epdb1.py

 > /root/epdb1.py(4)?()

 -> b = "bbb"

 (Pdb) n

 > /root/epdb1.py(5)?()

 -> c = "ccc"

 (Pdb) p b

‘bbb‘

 (Pdb)

‘bbb‘

 (Pdb) n

 > /root/epdb1.py(6)?()

 -> final = a + b + c

 (Pdb) p c

‘ccc‘

 (Pdb) p final

 *** NameError: <exceptions.NameError instance at 0x1551b710 >

 (Pdb) n

 > /root/epdb1.py(7)?()

 -> print final

 (Pdb) p final

‘aaabbbccc‘

 (Pdb)

使用 c 可以停止当前的 debug 使程序继续执行。如果在下面的程序中继续有 set_statement() 的申明,则又会重新进入到 debug 的状态,读者可以在代码 print final 之前再加上 set_trace() 验证。

清单 5. 停止 debug 继续执行程序

1

2

3

4

5

6

7

8

[[email protected] ~]#  python epdb1.py

 > /root/epdb1.py(4)?()

 -> b = "bbb"

 (Pdb) n

 > /root/epdb1.py(5)?()

 -> c = "ccc"

 (Pdb) c

 aaabbbccc

显示代码:在 debug 的时候不一定能记住当前的代码块,如要要查看具体的代码块,则可以通过使用 list 或者 l 命令显示。list 会用箭头 -> 指向当前 debug 的语句。

清单 6. debug 过程中显示代码

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

[[email protected] ~]#  python epdb1.py

 > /root/epdb1.py(4)?()

 -> b = "bbb"

 (Pdb) list

  1     import pdb

  2     a = "aaa"

  3     pdb.set_trace()

  4  -> b = "bbb"

  5     c = "ccc"

  6     final = a + b + c

  7     pdb.set_trace()

  8     print final

 [EOF]

 (Pdb) c

 > /root/epdb1.py(8)?()

 -> print final

 (Pdb) list

  3     pdb.set_trace()

  4     b = "bbb"

  5     c = "ccc"

  6     final = a + b + c

  7     pdb.set_trace()

  8  -> print final

 [EOF]

 (Pdb)

在使用函数的情况下进行 debug

清单 7. 使用函数的例子

1

2

3

4

5

6

7

8

9

10

11

import pdb

 def combine(s1,s2):      # define subroutine combine, which...

    s3 = s1 + s2 + s1    # sandwiches s2 between copies of s1, ...

    s3 = ‘"‘ + s3 +‘"‘   # encloses it in double quotes,...

    return s3            # and returns it.

 a = "aaa"

 pdb.set_trace()

 b = "bbb"

 c = "ccc"

 final = combine(a,b)

 print final

如果直接使用 n 进行 debug 则到 final=combine(a,b) 这句的时候会将其当做普通的赋值语句处理,进入到 print final。如果想要对函数进行 debug 如何处理呢 ? 可以直接使用 s 进入函数块。函数里面的单步调试与上面的介绍类似。如果不想在函数里单步调试可以在断点处直接按 r 退出到调用的地方。

清单 8. 对函数进行 debug

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

[[email protected] ~]# python epdb2.py

 > /root/epdb2.py(10)?()

 -> b = "bbb"

 (Pdb) n

 > /root/epdb2.py(11)?()

 -> c = "ccc"

 (Pdb) n

 > /root/epdb2.py(12)?()

 -> final = combine(a,b)

 (Pdb) s

 --Call--

 > /root/epdb2.py(3)combine()

 -> def combine(s1,s2):      # define subroutine combine, which...

 (Pdb) n

 > /root/epdb2.py(4)combine()

 -> s3 = s1 + s2 + s1    # sandwiches s2 between copies of s1, ...

 (Pdb) list

  1     import pdb

  2

  3     def combine(s1,s2):      # define subroutine combine, which...

  4  ->     s3 = s1 + s2 + s1    # sandwiches s2 between copies of s1, ...

  5         s3 = ‘"‘ + s3 +‘"‘   # encloses it in double quotes,...

  6         return s3            # and returns it.

  7

  8     a = "aaa"

  9     pdb.set_trace()

 10     b = "bbb"

 11     c = "ccc"

 (Pdb) n

 > /root/epdb2.py(5)combine()

 -> s3 = ‘"‘ + s3 +‘"‘   # encloses it in double quotes,...

 (Pdb) n

 > /root/epdb2.py(6)combine()

 -> return s3            # and returns it.

 (Pdb) n

 --Return--

 > /root/epdb2.py(6)combine()->‘"aaabbbaaa"‘

 -> return s3            # and returns it.

 (Pdb) n

 > /root/epdb2.py(13)?()

 -> print final

 (Pdb)

在调试的时候动态改变值 。在调试的时候可以动态改变变量的值,具体如下实例。需要注意的是下面有个错误,原因是 b 已经被赋值了,如果想重新改变 b 的赋值,则应该使用! B。

清单 9. 在调试的时候动态改变值

1

2

3

4

5

6

7

8

9

[[email protected] ~]# python epdb2.py

 > /root/epdb2.py(10)?()

 -> b = "bbb"

 (Pdb) var = "1234"

 (Pdb) b = "avfe"

 *** The specified object ‘= "avfe"‘ is not a function

 or was not found along sys.path.

 (Pdb) !b="afdfd"

 (Pdb)

pdb 调试有个明显的缺陷就是对于多线程,远程调试等支持得不够好,同时没有较为直观的界面显示,不太适合大型的 python 项目。而在较大的 python 项目中,这些调试需求比较常见,因此需要使用更为高级的调试工具。

pdb命令清单:

1)进入命令行Debug模式,python -m pdb xxx.py

2)h:(help)帮助

3)w:(where)打印当前执行堆栈

4)d:(down)执行跳转到在当前堆栈的深一层(个人没觉得有什么用处)

5)u:(up)执行跳转到当前堆栈的上一层

6)b:(break)添加断点

b 列出当前所有断点,和断点执行到统计次数

b line_no:当前脚本的line_no行添加断点

b filename:line_no:脚本filename的line_no行添加断点

b function:在函数function的第一条可执行语句处添加断点

7)tbreak:(temporary break)临时断点

在第一次执行到这个断点之后,就自动删除这个断点,用法和b一样

8)cl:(clear)清除断点

cl 清除所有断点

cl bpnumber1 bpnumber2... 清除断点号为bpnumber1,bpnumber2...的断点

cl lineno 清除当前脚本lineno行的断点

cl filename:line_no 清除脚本filename的line_no行的断点

9)disable:停用断点,参数为bpnumber,和cl的区别是,断点依然存在,只是不启用

10)enable:激活断点,参数为bpnumber

11)s:(step)执行下一条命令

如果本句是函数调用,则s会执行到函数的第一句

12)n:(next)执行下一条语句

如果本句是函数调用,则执行函数,接着执行当前执行语句的下一条。

13)r:(return)执行当前运行函数到结束

14)c:(continue)继续执行,直到遇到下一条断点

15)l:(list)列出源码

l 列出当前执行语句周围11条代码

l first 列出first行周围11条代码

l first second 列出first--second范围的代码,如果second<first,second将被解析为行数

16)a:(args)列出当前执行函数的函数

17)p expression:(print)输出expression的值

18)pp expression:好看一点的p expression

19)run:重新启动debug,相当于restart

20)q:(quit)退出debug

21)j lineno:(jump)设置下条执行的语句函数

只能在堆栈的最底层跳转,向后重新执行,向前可直接执行到行号

22)unt:(until)执行到下一行(跳出循环),或者当前堆栈结束

23)condition bpnumber conditon,给断点设置条件,当参数condition返回True的时候bpnumber断点有效,否则bpnumber断点无效

注意:

1:直接输入Enter,会执行上一条命令;

2:输入PDB不认识的命令,PDB会把他当做Python语句在当前环境下执行;

参考链接:

1. Python 调试器之pdb https://www.cnblogs.com/xiaohai2003ly/p/8529472.html

2. Python 代码调试技巧 https://www.ibm.com/developerworks/cn/linux/l-cn-pythondebugger/index.html

原文地址:https://www.cnblogs.com/ArsenalfanInECNU/p/10096128.html

时间: 2024-07-29 12:17:08

使用pdb模块调试Python的相关文章

使用pdb模块调试Python脚本

在Python中,语法错误可以被Python解释器发现,但逻辑上错误或变量使用错误却不容易发现,如果结果没有符合预期,则需要调试,一个很好的调试工具:Python自带的pdb模块.pdb是Python自带的调试模块.使用pdb模块可以为脚本设置断点.单步执行.查看变量值等. pdb可以用命令行参数的方式启动,也可以使用import 将其导入后再使用. >>> dir(pdb) ['Pdb', 'Repr', 'Restart', 'TESTCMD',.....,'re', 'run',

Linux下使用pdb简单调试python程序

python自带调试工具库:pdb # -*- coding:utf-8 -*- def func(num): s = num * 10 return s if __name__ == '__main__': print 'debug starting...' print '*' * 10 print 'debug ending-' num = 100 s = func(num) print s 在python文件中不引用pdb库,可以在执行python文件的时候,加上参数: python -m

用PDB库调试Python程序

用pdb调试有多种方式可选: 1. 命令行启动目标程序,加上-m参数,这样调用myscript.py的话断点就是程序的执行第一行之前python -m pdb myscript.py 2 在程序中间插入一段程序,相对于在一般IDE里面打上断点然后启动debug,不过这种方式是hardcode的 import pdbpdb.set_trace()b = 2c = a + bprint (c)然后正常运行脚本,到了pdb.set_trace()那就会定下来,就可以看到调试的提示符(Pdb)了 常用的

[转] 用PDB库调试Python程序

Python自带的pdb库,发现用pdb来调试程序还是很方便的,当然了,什么远程调试,多线程之类,pdb是搞不定的. 用pdb调试有多种方式可选: 1. 命令行启动目标程序,加上-m参数,这样调用myscript.py的话断点就是程序的执行第一行之前 python -m pdb myscript.py 2. 在Python交互环境中启用调试 >>> import pdb >>> import mymodule >>> pdb.run(‘mymodule

Linux通过使用pdb简单调试python计划

python自带的调试工具库:pdb # -*- coding:utf-8 -*- def func(num): s = num * 10 return s if __name__ == '__main__': print 'debug starting...' print '*' * 10 print 'debug ending-' num = 100 s = func(num) print s 在python文件里不引用pdb库,能够在运行python文件的时候,加上參数: python -

最简单方法远程调试Python多进程子程序

Python 2.6新增的multiprocessing,即多进程,给子进程代码调试有点困难,比如python自带的pdb如果直接在子进程代码里面启动会抛出一堆异常,原因是子进程的stdin/out/err等文件都已关闭,pdb无法调用.据闻winpdb.Wing IDE的调试器能够支持这样的远程调试,但似乎过于重量级(好吧前者比后者要轻多了,但一样要wxPython的环境,再说pdb的灵活可靠它们难以比拟). 其实只需稍作改动即可用pdb继续调试子进程的代码,思路来自这个博客:子进程的stdi

Python 中的 pdb 模块

PYTHON 代码,尤其是别人写的代码看不懂.怎么办? 其实PYTHON中也提供了类似于C语言中用于debug 的 gdb.它叫做pdb.结合本人自己的学习,进行简单的举例,以做备忘和补偿学习. 首先参考资料: 1.http://web.stanford.edu/class/physics91si/2013/handouts/Pdb_Commands.pdf 2.https://docs.python.org/2/library/pdb.html 以 shadowsocks 的 local.py

使用 pdb 进行调试

使用 pdb 进行调试 pdb 是 python 自带的一个包,为 python 程序提供了一种交互的源代码调试功能,主要特性包括设置断点.单步调试.进入函数调试.查看当前代码.查看栈片段.动态改变变量的值等.pdb 提供了一些常用的调试命令,详情见表 1. 表 1. pdb 常用命令 命令 解释 break 或 b 设置断点 设置断点 continue 或 c 继续执行程序 list 或 l 查看当前行的代码段 step 或 s 进入函数 return 或 r 执行代码直到从当前函数返回 ex

调试Python的方式

调试Python有如下几种方式: 1 使用print语句 2 使用IDE的debuggers 3 使用命令行调试器pdb,这是Python的一个标准库,类似gdb 4 使用-i命令行选项.在使用命令行运行Python文件的时,可使用如下命令:python -i 文件名.py,这样就会进入交互模式,你可以打印查看变量 原文地址:https://www.cnblogs.com/chaoguo1234/p/9126075.html