正常的函数如下所示:
def calc_sum(*args): ax = 0 for n in args: ax = ax + n return ax
但如果函数返回值是一个函数,示例如下:
def lazy_sum(*args): def sum(): ax = 0 for n in args: ax = ax + n return ax return sum
可以看出被返回的函数不带参数,但是它可以引用外部函数如lazy_sum的参数,返回函数名,该函数不立即执行,而是等调用时,使用函数名+()再执行,同样的参数调用返回的函数都是一个新的函数。
调用示例如下:
>>> f = lazy_sum(1, 3, 5, 7, 9) >>> f <function lazy_sum.<locals>.sum at 0x101c6ed90>
可以看出直接调用lazy_sum(1,3,5,7,9)只会返回求和函数sum,但是返回的函数中又包含了调用时的参数、变量,如1,3,5,7,9,因此使用f()调用时得到25的结果,这种结构称为闭包。对于内部函数而言,引用了外层函数的变量,这种就属于闭包。
>>> f() 25
其实也可以有办法立即执行,因为return返回了f2(),而不是f2,如下例:
def f1(x):
k=200
def f2():
nonlocal x,k
x*=x
k*=k
print('k'+str(k))#输出40000
print('x'+str(x))#输出10000
f2()
print('k'+str(k))#输出40000
print('x'+str(x))#输出10000
f1(100)
上例还展示了一种情况,内部函数试图改变闭包函数的变量值,但是这种做法是不被允许的,会报错,但是上例中使用了nonlocal关键字,定义了与闭包函数的变量同名的两个变量(x,k均为闭包变量),因此在内部函数作用域内可以对这两个值进行改变,在闭包作用域中,这两个值也发生了改变。
输出如下:
k200 x100 10000 40000
返回函数不仅可以引用外部函数的参数,而且还可以引用外部函数内部定义的局部变量,因此如果返回函数中引用的变量发生变化,会导致出问题,如下例所示:
def count(): fs = [] for i in range(1, 4): def f(): return i*i fs.append(f) return fs f1, f2, f3 = count()
该例返回了一个函数列表,含有三个函数,调用时因为返回的函数并没有立即执行,而是等到调用了f()才执行,所以实际结果是
>>> f1() 9 >>> f2() 9 >>> f3() 9
就是因为等三个函数都返回时,闭包变量i变成了3,所以最终结果都是9,因此返回函数不能引用任何后续会发生变化的变量。
如果一定要引用循环变量,那么需要再创建一个函数A,循环变量放在该函数A的外围,用循环变量作为该函数A的参数,而该函数A内部返回函数B,引用函数A的参数,此时因为外部循环得到的值是通过参数输入到函数A中,该参数已经固定,不会变化。