7.12 访问闭包中定义的变量

    通常来讲,闭包的内部变量对于外界来讲是完全隐藏的。但是,你可以通过编写访问函数并将其作为函数属性绑定到闭包上来实现这个目的。例如:

    下面是使用的例子:

    1. >>> f = sample()
    2. >>> f()
    3. n= 0
    4. >>> f.set_n(10)
    5. >>> f()
    6. n= 10
    7. >>> f.get_n()
    8. 10

    为了说明清楚它如何工作的,有两点需要解释一下。首先, 声明可以让我们编写函数来修改内部变量的值。其次,函数属性允许我们用一种很简单的方式将访问方法绑定到闭包函数上,这个跟实例方法很像(尽管并没有定义任何类)。

    下面是一个交互式会话来演示它是如何工作的:

    1. >>> s = Stack()
    2. >>> s
    3. <__main__.ClosureInstance object at 0x10069ed10>
    4. >>> s.push(10)
    5. >>> s.push(20)
    6. >>> s.push('Hello')
    7. >>> len(s)
    8. 3
    9. >>> s.pop()
    10. 'Hello'
    11. >>> s.pop()
    12. >>> s.pop()
    13. 10
    14. >>>

    有趣的是,这个代码运行起来会比一个普通的类定义要快很多。你可能会像下面这样测试它跟一个类的性能对比:

    如果这样做,你会得到类似如下的结果:

    1. >>> from timeit import timeit
    2. >>> # Test involving closures
    3. >>> s = Stack()
    4. >>> timeit('s.push(1);s.pop()', 'from __main__ import s')
    5. 0.9874754269840196
    6. >>> # Test involving a class
    7. >>> s = Stack2()
    8. >>> timeit('s.push(1);s.pop()', 'from __main__ import s')
    9. >>>

    Raymond Hettinger对于这个问题设计出了更加难以理解的改进方案。不过,你得考虑下是否真的需要在你代码中这样做,而且它只是真实类的一个奇怪的替换而已,例如,类的主要特性如继承、属性、描述器或类方法都是不能用的。并且你要做一些其他的工作才能让一些特殊方法生效(比如上面 中重写过的 实现。)

    最后,你可能还会让其他阅读你代码的人感到疑惑,为什么它看起来不像一个普通的类定义呢?(当然,他们也想知道为什么它运行起来会更快)。尽管如此,这对于怎样访问闭包的内部变量也不失为一个有趣的例子。

    总体上讲,在配置的时候给闭包添加方法会有更多的实用功能,比如你需要重置内部状态、刷新缓冲区、清除缓存或其他的反馈机制的时候。