PDB:变量可以打印但未定义

时间:2017-07-25 19:33:40

标签: python python-2.7 pdb

看起来很奇怪。打印名称为 classes 的变量,但在尝试执行filter(...)构造时未定义。

这是一段代码:

def start(self, tag, attrib):
    classes = attrib[self._CLASS_ATTR] if self._CLASS_ATTR in attrib else None

    if tag == self._TAG_P:
        p = self._doc.add_paragraph('')

        self._cur_p = p

        if classes is not None:
            alignments = [self._left_align, self._center_align, self._right_align]
            import pdb; pdb.set_trace()
            alignments = filter(lambda x: partial(x.is_in, classes), alignments)
            if len(alignments) > 0:
                p.alignment = alignments[0].get()

            assert len(alignments) < 2

Pdb在它休息时停止。当我尝试执行filter()时:

(Pdb) print filter(lambda x: partial(x.is_in, classes), alignments)
*** NameError: global name 'classes' is not defined

可是:

(Pdb) print classes
center title
(Pdb) classes
u'center title'

为什么filter(...)指令无法正常执行?

让我们用简短的代码重现它:

from functools import partial

def f():
    classes = 'my_classes'

    def my_bool(obj, _):
        return True

    if classes is not None:
        import pdb; pdb.set_trace() # point a
        alignments = filter(lambda x: my_bool(x, classes), ['1', '2', '3'])
        import pdb; pdb.set_trace() # point b
        pass

f()

...

(Pdb) filter(lambda x: my_bool(x, classes), ['1', '2', '3'])
*** NameError: global name 'my_bool' is not defined

但是,点c中pdb的命令a(继续)不会产生异常。

1 个答案:

答案 0 :(得分:2)

pdb是一个eval循环。一个eval循环基本上将你写入提示的内容逐行地带到eval(...)它。这意味着它不会在定义的函数(lambdas)中绑定闭包范围的变量。 eval(这是一个函数)有自己的范围,不参与您正在评估的闭包。

您可以在此示例代码中看到等效问题:

def f():
    x = 1
    return eval('lambda: x')

>>> f()()
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
  File "<string>", line 1, in <lambda>
NameError: name 'x' is not defined

(不幸)解决方法是预先定义任何lambdas并在pdb表达式中使用它们。