一.闭包的定义:
创新互联公司2013年至今,先为荆门等服务建站,荆门等地企业,进行企业商务咨询服务。为荆门企业网站制作PC+手机+微官网三网同步一站式服务解决您的所有建站问题。
在一个函数的内部,再定义一个函数(内部函数)。这个内部函数引用了外部函数的变量,并且外部函数返回这个内部函数, 我们把这个使用外部函数变量的内部函数称为 闭包 。
简而言之, 闭包就是能够读取外部函数内的变量的函数。
例如:
形成闭包的两个条件:
二.闭包的用途
① 可以读取函数内部的变量
② 将一些变量的值始终保存到内存中
1.读取函数内部的变量
在一般情况下,在函数外部我们是不能访问到函数内部的变量的。但是, 有时想要在函数外部能够访问到函数内部的变量,那么就可以使用闭包。
例如:
上面的代码可以看出,print(a)会抛异常NameError: name 'a' is not defined。在函数f1的外面无法访问它的变量的。
在函数f1里面定义一个闭包函数就可以访问到了
例如:
2.将一些变量的值始终保存到内存中
运行结果:
通过上面的输出结果可以看出闭包保存了外部函数内的变量n1的值1,每次执行闭包都是在n1 = 1 基础上进行计算的。
三.闭包的缺点
1. 由于闭包会使得函数中的变量都被保存在内存中,会增加 内存消耗 ,所以不能滥用闭包,否则会造成程序的性能问题,可能导致内存泄露
2. 闭包无法改变外部函数局部变量指向的内存地址
3. 返回闭包时,返回函数不要引用任何循环变量,或者后续会发生变化的变量
四.判断一个函数是否是闭包
判断一个函数是不是闭包,可以查看它的 closure 属性。如果该函数是闭包,查看该属性将会返回一个cell对象组成的tuple。如果我们分别对每个cell对象查看其cell_contents属性,返回的内容就是闭包引用的自由变量的值。
运行结果:
闭包的__closure__方法,可以展示出闭包储存了外部函数的两个变量,cell的内存地址是什么,在cell里面储存的对象类型是int,这个int储存的内存地址是什么。
闭包的__closure__方法,可以查看每个cell对象的内容
运行结果:
cell_contents解释了局部变量在脱离函数后仍然可以在函数之外被访问的原因,因为变量被存储在cell_contents中了。
在python3.0之前的版本中, 闭包只能够访问而不能修改外部变量(非全局), 所以, 通常为了方便修改, 我们就使用可变对象来通过修改对象内部的一些引用来达到间接修改的目的, 因此, 通常的数字类型, 字符串类型等不可变类型不可以作为用于修改的自由变量(闭包中引用的外部变量)
在python3.0及以后版本中, 通过nonlocal关键字解决了这个问题, 并增强了闭包. 在之前的版本中, 外部变量实际上是会放入内部函数(闭包)的local这个名称空间中的, 所以, 在之前版本中, 我们在闭包内试图修改一个外部变量的时候, 往往会得到一个"在引用之前没有赋值"的一个错误. 3.0之后, 给要在闭包内修改的变量加上(在闭包内修改之前)nonlocal的声明, 然后, 在下面的代码中修改就可以了....
这个用法和global关键字解决函数内修改全局变量有异曲同工之妙...
因为python具有late binding的机制——闭包中内部函数的值只有在被调用时才会确定,等到f1,f2,f3调用时,此时闭包中f()函数的i已经等于3了,于是所有结果等于9.
如果想得到你要的结果,就得提前把i作为参数传入,把原先的def f():所在行修改为def f(i=i)即可,其中等号左边的i是形参,等号右边的i是for in 循环中对应的i
def count():
fs = []
for i in range(1,4):
def f(i=i):
return i**2
fs.append(f)
return fs
f1,f2,f3 = count()
print(f1())
print(f2())
print(f3())
输出
1
4
9
在python中,函数可以被嵌套定义,也就是说,函数中可以定义函数。该函数还可以将其内部定义的函数作为返回值返回。
闭包的定义:一般来说,我们可以认为,如果一个函数可以读取其他函数中的局部变量,那么它们就构成了闭包。
注意 :闭包的定义不是特别清晰,但大体上的意思是这样的。
我们知道,普通的函数是可以使用全局变量的
类似的,函数中定义的函数,也是可以使用外部函数的变量的。因此,满足了函数读取了其他函数局部变量的这一条件,他们因此构成了闭包。
在闭包的使用中,我们可以先给外部的函数赋予不同的局部变量,然后再调用其中内部的函数时,就可以读取到这些不同的局部变量了。
外部变量的使用 在普通函数中,虽然可以直接使用全局变量,但是不可以直接修改全局变量。从变量的作用域来说,一旦你尝试修改全局变量,那么就会尝试创建并使用一个同名的局部变量。因此,如果你需要在普通函数中修改全局变量,需要使用global
同样的,如果你希望通过定义在内部的函数去修改其外部函数的变量,那么必须使用nonlocal