gdb中是否有等同于disp
的pdb?
E.g。当我使用gdb调试C时,我可以通过输入代码在每个'step'上打印变量:
disp var
当我使用pdb调试python时,我想要类似的功能,但disp
似乎不存在,python pdb文档似乎没有提供替代 - 但它似乎是一个奇怪的遗漏?
答案 0 :(得分:3)
下面的代码使用Python内省功能向PDB模块0添加两个新命令 只需将给定的函数及其调用放在一个单独的模块中,然后在开始调试之前导入该模块 - 您应该使用'disp'和'undisp'命令将监视器添加和收回变量。
它通过monkeypatching Python的pdb模块工作,它是用纯python编写的。
# -*- coding: utf-8 -*-
def patch_pdb():
import pdb
def wrap(func):
def new_postcmd(self, *args, **kw):
result = func(self, *args, **kw)
if hasattr(self, "curframe") and self.curframe and hasattr(self, "watch_list"):
for arg in self.watch_list:
try:
print >> self.stdout, "%s: %s"% (arg, self._getval(arg)) + ", ",
except:
pass
self.stdout.write("\n")
return result #func(self, *args, **kw)
return new_postcmd
pdb.Pdb.postcmd = wrap(pdb.Pdb.postcmd)
def do_disp(self, arg):
if not hasattr(self, "watch_list"):
self.watch_list = []
self.watch_list.append(arg)
pdb.Pdb.do_disp = do_disp
def do_undisp(self, arg):
if hasattr(self, "watch_list"):
try:
self.watch_list.remove(arg)
except:
pass
pdb.Pdb.do_undisp = do_undisp
patch_pdb()
if __name__ == "__main__":
# for testing
import pdb; pdb.set_trace()
a = 0
for i in range(10):
print i
a += 2
不幸的是,我只能让它显示变量的状态,就像之前执行最后一个命令一样。 (我尝试了一下,但monkeypatching bdb模块,这是Pdb的基础似乎也没有工作)。您可以尝试更改由wrap
修饰的pdb.Pdb,bdb.Bdb或cmd.Cmd中的方法,以查找在调试的帧状态更改后调用的方法。
答案 1 :(得分:3)
您可以设置一些可以为您执行此操作的别名:
alias n next;; p var
alias s step;; p var
打印整个变量名列表留给读者练习。不幸的是,这样做意味着当您向调试器发送一个空行时,它执行的“最后一个命令”是p var
,而不是n
。如果你想修复它,那么你可以使用这个有点hacky的Pdb命令集:
!global __stack; from inspect import stack as __stack
!global __Pdb; from pdb import Pdb as __Pdb
!global __pdb; __pdb = [__framerec[0].f_locals.get("pdb") or __framerec[0].f_locals.get("self") for __framerec in __stack() if (__framerec[0].f_locals.get("pdb") or __framerec[0].f_locals.get("self")).__class__ == __Pdb][-1]
alias s step;; p var;; !__pdb.lastcmd = "!__pdb.cmdqueue.append('s')"
alias n next;; p var;; !__pdb.lastcmd = "!__pdb.cmdqueue.append('n')"
答案 2 :(得分:1)
在pdb调试期间,您可以键入普通的python代码,超出单字母命令 - 所以只需使用print var
就可以了。