Damon

宏愿纵未了 奋斗总不太晚

0%

Python闭包

Python基础

Python中,函数也是一个对象,而且函数对象可以被赋值给变量,所以,通过变量也能调用该函数。
例如:

1
2
3
4
5
def now():
print('2017-11-06')

f = now
f() # 打印 2017-11-06

理解什么是闭包?

Python 中,闭包的概念是:
1.首先是个嵌套函数
2.内层函数使用了外层函数的变量或者参数
3.外层函数把内层函数当做返回值进行返回

一段简单的闭包代码示例

1
2
3
4
5
6
7
8
9
10
11
12
def func_outer(): # 外部函数

temp = 666 # 外部函数的变量

def func_inner(): # 嵌套的内部函数
print(temp) # 使用了外部函数的变量

return func_inner # 返回内部函数


test = func_outer()
test() # 打印666

由以上可以看出,test变量其实就是返回的func_innter函数对象,test()就是通过这个变量来调用func_innter函数.

闭包的注意点:

请思考一下闭包中下面的几种情况

1.在内层函数修改了外层函数的变量
2.在闭包中,引用了一个可能会发生变化的变量

先看第一种情况,对代码进行修改一下

1
2
3
4
5
6
7
8
9
10
11
12
13
def func_outer():
temp = 666
def func_inner():
temp = 888 # 0.在内层函数中修改了外层函数的变量的值

print(temp) # 1
func_inner() #2. 在外部就执行了内部函数
print(temp) # 3.

return func_inner


func_outer()

在注释3处,打印的值还是666,这个问题比较简单,因为在注释0代码处相当于重新声明了一个局部变量temp,所以这只是一个作用域的问题(如果你是编程新手的话请打开你的搜索引擎了解作用域的知识)
其实很简单,就四个大字: 就近原则

如果要修改外部函数的temp变量使用关键字nonlocal声明:
注释0处代码改为

1
2
nonlocal temp
temp = 888

第二种情况,先看代码:

1
2
3
4
5
6
7
8
9
10
11
def func_outer():
temp = 666
def func_inner():
print(temp)

# 修改了变量的值
temp = 888
return func_inner

test = func_outer()
test()

上面代码在声明内部函数func_inner()之后修改了temp变量的值,然而打印结果却是888.

要想搞懂原因的话可以先写一个小测试

1
2
def test_func():
print(nondefine)

上面函数打印了一个未被声明过的变量,但是,如果就这样运行的话程序时没有问题的,照样能正常运行.而当我们调用这个函数的时候test_func(),再次运行程序就报错了.报错原因

NameError: name ‘nondefine’ is not defined

上面的测试证明了当一个函数被调用时,它内部变量对应的值才会被确定,这就能解释上面为什么会打印888了.