浅析Python闭包

一、什么是闭包
python

在介绍闭包概念前,咱们先来看一段简短的代码数组

def sum_calc(*args):
    def wrapper():
        sum = 0
        for n in args:
            sum += n;
        return sum
    return wrapper

很显然,这段代码定义了一个名为sum_calc的函数,但和定义的普通函数不一样的是这个函数体的内部又定义了一个名为wrapper的函数,而且sum_calc函数的返回值是内部定义wrapper函数。闭包

如今咱们开始来调用sum_calc函数,看看会出现哪些有趣的事情app

>>> f = sum_calc(1, 2, 3, 4, 5)
>>> f
<function sum_calc.<locals>.wrapper at 0x0000025693AC2D30>
>>> f() 15

从运行结果来看,当咱们调用sum_calc时,返回的并非求和结果,而是内部定义的求和函数。函数

继续调用sum_calc返回函数时,才真正计算出求和的结果。spa

当咱们继续调用一次sum_calc,传入相同参数code

>>> f1 = sum_calc(1, 2, 3, 4, 5)
>>> f2 = sum_calc(1, 2, 3, 4, 5)
>>> f1 == f2
False
>>> f1()
15
>>> f2()
15

每次调用sum_calc,即便传入相同的参数,两次返回的对象不一样,且f1()和f2()的结果互不影响。对象

在这个例子中,咱们在sum_calc函数中定义的wrapper函数,wrapper函数能够引用外部函数sum_calc的参数和局部变量,当sum_calc返回函数wrapper时,相关参数保存在返回的函数中,能够被返回的函数继续使用,咱们把这种状况称为“闭包”。blog

参考维基百科,对“闭包”进行更严谨的解释:io

在计算机科学中,闭包(英语:Closure),又称词法闭包(Lexical Closure)或函数闭包(function closures),是引用了自由变量的函数。这个被引用的自由变量将和这个函数一同存在,即便已经离开了创造它的环境也不例外。因此,有另外一种说法认为闭包是由函数和与其相关的引用环境组合而成的实体。闭包在运行时能够有多个实例,不一样的引用环境和相同的函数组合能够产生不一样的实例。

二、闭包陷阱

def my_fun():
    fs = []
    for i in range(0, 3):
        def f():
             return i*i
        fs.append(f)
    return fs
f1, f2, f3 = count()

在上面的例子中,每次循环,都建立了一个新的函数,而后,把建立的3个函数都放在列表中返回。

咱们可能认为,f1()、f2()、f3()结果是0、一、4,但实际结果是4

>>> f1()
4
>>> f2()
4
>>> f3()
4

这个例子能够说是典型的错误使用闭包的案例,my_fun返回的并非一个闭包函数,而是一个包含三个闭包函数的一个列表。

在返回闭包列表fs以前for循环的变量的值已经发生改变了,循环内闭包函数仅声明了本身计算方式,并不会当即使用当前i的值进行计算。只有在真正调用闭包函数时,才会真正的执行闭包函数内的计算,而此时存放的i的值已是2,因此f1()、f2()、f3()的结果是4而不是我在以前期待的0、一、4。

通过上面的分析,咱们得出下面一个重要的经验:返回闭包中不要引用任何循环变量,或者后续会发生变化的变量。

正确写法

def my_fun(*args):
    L = []
    for i in range(3):
        def wrapper(_i = i):
            return _i * _i
        L.append(wrapper)
    return L    

三、闭包实现机制 

闭包比普通的函数多了一个 __closure__ 属性,该属性记录着自由变量内容。当闭包被调用时,系统就会根据该地址找到对应的自由变量,完成总体的函数调用

还以 sum_calc() 为例,当其被调用时,能够经过 __closure__ 属性获取自由变量(也就是程序中的 args参数)内容,例如:

def sum_calc(*args):
    def wrapper():
        sum = 0
        for n in args:
            sum += n;
        return sum
    return wrapper

输出结果:

>>> f1 = sum_calc(1, 2, 3, 4, 5)
>>> f1.__closure__
(<cell at 0x0000025693A5E4C0: tuple object at 0x0000025693AE5400>,)

>>> f1.__closure__[0].cell_contents
(1, 2, 3, 4, 5)

 能够看到,显示的内容是一个tuple类型,这就是f1中自由变量args的值。还能够看到,__closure__ 属性的类型是一个元组,这代表闭包能够支持多个自由变量的形式。

相关文章
相关标签/搜索