Python基础之闭包

Python016

Python基础之闭包,第1张

一.闭包的定义:

在一个函数的内部,再定义一个函数(内部函数)。这个内部函数引用了外部函数的变量,并且外部函数返回这个内部函数, 我们把这个使用外部函数变量的内部函数称为 闭包

简而言之, 闭包就是能够读取外部函数内的变量的函数。

例如:

形成闭包的两个条件:

二.闭包的用途

可以读取函数内部的变量

将一些变量的值始终保存到内存中

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中了。

在Python语言中,可以在函数中定义函数。 这种在函数中嵌套定义的函数也叫内部函数。我们来看下面的代码:

上述代码中,定义了函数greet,在函数greet内部又定义了一个函数inner_func, 并调用该函数打印了一串字符。

我们可以看到,内部函数inner_func的定义和使用与普通函数基本相同。需要注意的是变量的作用域,在上述代码中,函数参数name对于全局函数greet是局部变量,对内部函数inner_func来说则是非局部变量。内部函数对于非局部变量的访问规则类似于标准的外部函数访问全局变量。

从这个例子我们还可以看到内部函数的一个作用,就是通过定义内部函数的方式将一些功能隐藏起来,防止外部直接调用。常见的场景是,在一个复杂逻辑的函数中,将一些小的任务定义成内部函数,然后由这个外层函数使用,这样可以使代码更为清晰,易于维护。这些内部函数只会在这个外层函数中使用,不能被其他函数或模块使用。

在Python语言中, 函数也是对象,它可以被创建、赋值给变量,或者作为函数的返回值。我们来看下面这个例子。

在上述代码中,在函数gen_greet内部定义了inner_func函数,并返回了一个inner_func函数对象。外部函数gen_greet返回了一个函数对象,所以像gen_greet这样的函数也叫工厂函数。

在内部函数inner_func中,使用了外部函数的传参greet_words(非局部变量),以及函数的参数name(局部变量),来打印一个字符串。

接下来,调用gen_greet("Hello")创建一个函数对象say_hello,紧接着调用say_hello("Mr. Zhang"),输出的结果为:Hello, Mr. Zhang!

同样的,调用gen_greet("Hi")创建一个函数对象say_hi,调用say_hello("Mr. Zhang"),输出的结果为:Hi,Tony!

我们可以发现,gen_greet返回的函数对象具有记忆功能,它能够把所需使用的非局部变量保存下来,用于后续被调用的时候使用。这种保存了非局部变量的函数对象被称作闭包(closure)。

那么闭包是如何实现的呢?其实并不复杂,函数对象中有一个属性__closure__,它就是在创建函数对象时用来保存这些非局部变量的。

__closure__属性是一个元组或者None类型。在上述代码中,我们可以通过下面方式查看:

函数的嵌套所实现的功能大都可以通过定义类的方式来实现,而且类是更加面向对象的代码编写方式。

嵌套函数的一个主要用途是实现函数的装饰器。我们看下面的代码:

在上述代码中,logger函数返回函数with_logging,with_logging则是打印了函数func的名称及传入的参数,然后调用func, 并将参数传递给func。其中的@wraps(func)语句用于复制函数func的名称、注释文档、参数列表等等,使得with_logging函数具有被装饰的函数func相同的属性。

代码中接下来用@logger对函数power_func进行修饰,它的作用等同于下面的代码:

可见,装饰器@符其实就是上述代码的精简写法。

通过了解了嵌套函数和闭包的工作原理,我们在使用过程中就能够更加得心应手了。