Python pdb(调试器)disp等价吗?

时间:2011-12-15 13:04:14

标签: python debugging pdb

gdb中是否有等同于disp的pdb?

E.g。当我使用gdb调试C时,我可以通过输入代码在每个'step'上打印变量:

disp var

当我使用pdb调试python时,我想要类似的功能,但disp似乎不存在,python pdb文档似乎没有提供替代 - 但它似乎是一个奇怪的遗漏?

3 个答案:

答案 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就可以了。