看起来很奇怪。打印名称为 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
(继续)不会产生异常。
答案 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表达式中使用它们。