返回函数
1)普通的函数返回一个值,返回函数则是把函数作为结果值返回
为什么要返回函数:不需要立刻求和,而是在后面的代码中,根据需要再计算,这样就先返回求和的函数。(这样迂回的手段,听起来就好像为了优化代码。。。)
def lazy_sum(*args):
def sum():
ax = 0
for n in args:
ax = ax + n
return ax
return sum
2)当我们调用lazy_sum(1, 3, 5, 7, 9)时,返回的并不是求和结果,而是求和函数:
>>> f = lazy_sum(1, 3, 5, 7, 9)
>>> f
<function lazy_sum.<locals>.sum at 0x101c6ed90>
因为lazy_sum()函数return sum
意思就是,无论lazy_sum(1, 3, 5, 7, 9)
括号里写成什么,它都不会先计算,而是先返回一个sum函数,这个时候参数以及被包含在里面了。f = lazy_sum(1, 3, 5, 7, 9)
意思是,将函数返回的sum这个函数对象赋值给了f,而不是调用了sum函数,所以>>> f
会返回一个类似内存地址一样的东西。
3)关于 f 和 f():函数对象VS函数调用,鉴于一开始也混淆了,所以还是要强行解释一波
把函数赋值给新的标识符或作为参数传递给新的函数,针对的都是函数对象本身,而不是函数的调用。
def func():
return "hello,world"
ref1 = func #将函数对象赋值给ref1 赋值的结果是函数 ,可调用
ref2 = func() #调用函数,将函数的返回值("hello,world"字符串)赋值给ref2 赋值的结果是str,不可调用
所以,回答上面关于f = lazy_sum(1, 3, 5, 7, 9)
的情况上面,可以写成print(lazy_sum(1,3,5,7,9)())
这个样子。
4)注意:当我们调用lazy_sum()
时,每次调用都会返回一个新的函数,即使传入相同的参数:
>>> f1 = lazy_sum(1,3,5,7,9)
>>> f2 = lazy_sum(1,3,5,7,9)
>>> print(f1==f2)
False
在函数lazy_sum中又定义了函数sum,并且,内部函数sum可以引用外部函数lazy_sum的参数和局部变量,当lazy_sum返回函数sum时,相关参数和变量都保存在返回的函数中,这种称为“闭包(Closure)”。