为何在exec内关闭闭包?

发布于 2021-01-29 17:45:58

在Python 2.6中,

>>> exec "print (lambda: a)()" in dict(a=2), {}
2
>>> exec "print (lambda: a)()" in globals(), {'a': 2}
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
  File "<string>", line 1, in <module>
  File "<string>", line 1, in <lambda>
NameError: global name 'a' is not defined
>>> exec "print (lambda: a).__closure__" in globals(), {'a': 2}
None

我希望它打印2两次,然后打印一个元组cell。与3.1中的情况相同。这是怎么回事?

关注者
0
被浏览
44
1 个回答
  • 面试哥
    面试哥 2021-01-29
    为面试而生,有面试问题,就找面试哥。

    当您将字符串传递给exec或时eval,它将在考虑全局变量或局部变量之前将该字符串编译为代码对象。所以当你说:

    eval('lambda: a', ...)
    

    它的意思是:

    eval(compile('lambda: a', '<stdin>', 'eval'), ...)
    

    没有办法compile知道这a是一个freevar,因此将其编译为全局引用:

    >>> c= compile('lambda: a', '<stdin>', 'eval')
    >>> c.co_consts[0]
    <code object <lambda> at 0x7f36577330a8, file "<stdin>", line 1>
    >>> dis.dis(c.co_consts[0])
      1           0 LOAD_GLOBAL              0 (a)
                  3 RETURN_VALUE
    

    因此,要使其发挥作用,您必须a输入全局变量而不是本地变量。

    是的,这有点狡猾。但是那是对的execeval我想…对于他们来说,它们应该不是很好。



知识点
面圈网VIP题库

面圈网VIP题库全新上线,海量真题题库资源。 90大类考试,超10万份考试真题开放下载啦

去下载看看