对于一些快速的
Python调试,我偶尔会抛出一个导入pdb; pdb.set_trace()行会让我进入调试器.非常便利.但是,如果我想调试一个可能运行很多次的循环,它会在一定程度上失去它的效果.我可以继续许多次,多次,但是有没有办法删除/忽略那个硬编码的断点,这样我就可以让它完成了?
我可以设置一个全局标志并有条件地运行它,但后来我失去了单行断点的’standalone-ness’,还需要为每个pdb.set_trace()另一个标志.
解决方法
我的另一个答案是快速破解,并不完美(将禁用所有对set_trace的调用).这是一个更好的解决方案.
我们定义一个包装pdb的模块.在其中,set_trace是一个可调用对象,维护一个被禁用的调用者列表(由filename / line-number标识).
# mypdb.py import inspect import sys try: import ipdb as PDB except ImportError: import pdb as PDB class SetTraceWrapper(object): def __init__(self): self.callers_disabled = set() self.cur_caller = None def __call__(self): self.cur_caller = self.get_caller_id() if self.cur_caller in self.callers_disabled: # already disabled for caller #print 'set_trace SKIPPED for %s' % ( self.cur_caller,) return #print 'set_trace BREAKING for %s' % ( self.cur_caller,) try: PDB.set_trace(sys._getframe().f_back) except TypeError: PDB.set_trace() def disable_current(self): #print 'set_trace disABLING %s' % ( self.cur_caller,) self.callers_disabled.add(self.cur_caller) def get_caller_id(self,levels_up = 1): f = inspect.stack()[levels_up + 1] return ( f[1],f[2] ) # filename and line number set_trace = SetTraceWrapper()
在您的代码中,请确保使用包装器:
import mypdb as pdb; pdb.set_trace()
如果要禁用当前的set_trace-calling-line,请执行以下操作:
pdb.set_trace.disable_current()
笔记:
>我个人更喜欢ipdb而不是pdb>使用pdb时,由于实际调用pdb.set_trace的函数位于包装器中,因此当前帧在断开时将在其中. up命令可以获得你想要的帧.如果使用ipdb,则不会发生这种情况(包装器中的实现确保在正确的位置中断).>当使用ipdb时,我发现它不一致地报告调用者帧行号.这意味着第一次执行pdb.set_trace.disable_current()时,它可能不成立.如果下次再次破坏,请再次执行 – 第二次持续.>通常,拥有自己的pdb包装器对其他事情也很有用.我有自己的set_trace包装器,如果不是sys.stdout.isatty(如果进程没有连接到终端,也不想将stdout重定向到文件/管道,你永远不想破坏).也就是说,拥有自己的pdb包装器并调用其set_trace而不是pdb是一个很好的做法.
版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。