Python的类闭包如何工作?

时间:2010-02-03 14:59:08

标签: python closures bytecode

如果我针对本地命名空间创建一个类,它究竟是如何工作的?例如:

>>> def foo():
...     i = 1
...     class bar(object):
...             j = i
...     return bar
... 
>>> dis(foo)
  2           0 LOAD_CONST               1 (1)
              3 STORE_DEREF              0 (i)

  3           6 LOAD_CONST               2 ('bar')
              9 LOAD_GLOBAL              0 (object)
             12 BUILD_TUPLE              1
             15 LOAD_CLOSURE             0 (i)
             18 BUILD_TUPLE              1
             21 LOAD_CONST               3 (<code object bar at 0xb74f8800, file "<stdin>", line 3>)
             24 MAKE_CLOSURE             0
             27 CALL_FUNCTION            0
             30 BUILD_CLASS         
             31 STORE_FAST               0 (bar)

  5          34 LOAD_FAST                0 (bar)
             37 RETURN_VALUE        

我特别感兴趣的是:

             15 LOAD_CLOSURE             0 (i)
             18 BUILD_TUPLE              1
             21 LOAD_CONST               3 (<code object bar at 0xb74f8800, file "<stdin>", line 3>)
             24 MAKE_CLOSURE             0
             27 CALL_FUNCTION            0
             30 BUILD_CLASS

我想我想知道的最重要的事情是制作什么功能然后调用?这个函数是闭包附加到类的函数,还是其他地方发生的?

1 个答案:

答案 0 :(得分:2)

全班主体,即

j = i

是一个代码对象,它在偏移量21处加载,然后通过CALL_FUNCTION在偏移量27处调用。然后,调用的结果(本地名称空间)与类名和基础一起使用以创建类。 BUILD_CLASS有三个参数,类似于type(name, bases, dict)函数:

  

返回一个新类型对象。这实际上是类语句的动态形式。名称字符串是类名,并成为名称属性;基元元组列出基类并成为基础属性; dict字典是包含类体定义的命名空间,并成为 dict 属性。

还有一篇非常详细的文章"Notes on the Python Class Statement"解释了课程创作是如何运作的。