python3闭包通俗解释_python通俗解说闭包

本文深入探讨了Python中的闭包概念,通过示例解释了闭包如何保留外部环境的引用,以及闭包在函数作用域中的特殊性。文章还介绍了闭包的常见陷阱,如在循环中定义闭包导致的意外行为,并展示了如何利用闭包实现函数修饰以增加额外功能。此外,文章提到了闭包在类与实例关系上的类比,并讨论了闭包在实际应用中的场景,如接口函数和修饰器的使用。
摘要由CSDN通过智能技术生成

通俗明白闭包

先来看看什么是闭包吧

闭包是引用了自由变量的函数。这个被引用的自由变量将和这个函数一同存在,纵然已经离开了缔造它的环境也不破例。以是,有另一种说法以为闭包是由函数和与其相关的引用环境组合而成的实体。闭包在运行时可以有多个实例,差别的引用环境和相同的函数组合可以发生差别的实例。

这句话闭包是由函数和与其相关的引用环境组合而成的实体,我以为已经能归纳综合闭包的概念了。下面看看剖析

先看一个最简朴的例子

def outer_func():

outer_list = []

def inner_func():

outer_list.append(1)

print out_list

return inner_func

func1 = outer_func()

func1()#[1]

func1()#[1,1]

func2 = outer_func()

func2()#[1]

func2()#[1,1]

这个例子说明闭包与一样平常的函数不一样,他拥有的“环境”是独一份的。其中的outer_list称为自由变量,既不是全局变量又不是内陆变量。

If a name is bound in a block, it is a local variable of that block. If a name is bound at the module level, it is a global variable. (The variables of the module code block are local and global.) If a variable is used in a code block but not defined there, it is a free variable.

这种特征类似 类与实例 的关系,函数outer_func就像是一个类,执行func1 = outer_func就像是建立了一个实例,而实例func1能够继续类的属性,这里也可以看作是继续oucter_func的环境。

下面我换一种写法(这种写法是sml的写法。local与in之间就是环境)

local

outer_list = []

in

def inner_func():

outer_list.append(1)

print out_list

end

函数outer_func将环境outer_list = []与函数inner_func捆绑在一起,它的作用仅此而已

下面为了加深明白,我们再看一个闭包陷阱

def outer_func():

func_list = []

for i in xrange(3):

def inner_func():

print i

func_list.append(inner_func)

return func_list

fun1,fun2,fun3 = outer_func()

fun1()#2

fun2()#2

fun3()#2

我们再来通过拆分环境和函数来剖析outer_func

#执行fun1,fun2,fun3 = outer_func()之后,执行fun1()之前的环境

local

func_list = [inner_func1, inner_func2,inner_func3]

i = 2 #"环境初始化"完成之后,i就是2

in

def inner_func():

print i

end

这样可以看出i显著是2,但下面稍加改动

def outer_func():

func_list = []

for i in xrange(3):

def inner_func(_i = i): #写入默认参数

print _i

func_list.append(inner_func)

return func_list

fun1,fun2,fun3 = outer_func()

fun1()#0

fun2()#0

fun3()#2

剖析上面的程序

#这里展示func_list中 第一个 inner_func(func1)的环境

local

func_list = [inner_func1, inner_func2,inner_func3]

i = 2#外部的i照样2

in

def inner_func(_i = 0):#对于inner_func1,_i=0,这里可以发现形参_i实时捕捉i=0时的值,看成默认参数

print _i

end

为什么这时刻func1中的i=0呢?这是由于inner_func有了参数_i,它能在程序执行func_list.append(inner_func)的时刻,

会建立相关”函数实例“,而该函数界说中有一个带默认值的形参_i,注重python可以指定一个变量作为函数参数的默认值,

因此它会在建立的时刻也记录下i此时的值,即0.

而不带参数的inner_func,它只会在outer_func完全运行竣事之后,读取外部环境中i的值,即i=2。

下面说说闭包的应用

修饰函数

能在不改动已有函数内部组织的同时,添加分外功效,如检错功效。

#闭包使得先执行wrapper函数再执行func,可以控制函数执行的先后。

def func_dec( func ):

def wrapper( *args ):

if len(args) < 2:

print "less argument"

else:

func( args )

return wrapper

@func_dec

def mySum(*args):

print sum( *args )

mySum(2)#"less argument"

mySum(1,2,3) #6

剖析一下

#以mySum(2)为例子

local

func = mySum

in

def wrapper( *args ):#args = 2

if len(args) < 2:

print "less argument"

else:

func( args )

end

这样看思绪应该清晰不少

这里的 mySum(2) 等价于 func_dec( mySum )(2),func_dec后面接了2个括号,实在也可以看出func_dec肯定返回一个函数。

之以是搞得这么贫苦,就是为了让使用mySum的时刻附带一个检测参数个数的功效,条件是不改变mySum原有代码。类似接口函数。

上面的说mySum(2) 等价于 func_dec( mySum )(2),由此会发生一些隐晦的bug

,看看下面的例子:

def counter( cls ):

obj_list = []

def wrapper( *args, **kwargs ):

new_obj = cls( *args, **kwargs )

obj_list.append( new_obj )

print "class: %s' object number is %d" % (cls.__name__, len(obj_list) )

return new_obj

return wrapper

@counter

class my_cls( object ):

STATIC_MEN = "static"

def __init__( self, *arg, **kwargs):

print self, arg, kwargs

print my_cls.STATIC_MEN

my_cls()#AttributeError: 'function' object has no attribute 'STATIC_MEN'

为什么会说’function’ object has no attribute ‘STATIC_MEN’呢?

首先确定语句失足的位置:print my_cls.STATIC_MEN

那为什么my_cls不存在属性STATIC_MEN呢?

这是由于使用闭包后(@语法糖),my_cls() = counter(my_cls)()

这里应该被做了类似重定向的操作(由于语法糖@counter的缘故), 此时my_cls不再是原来的class,

执行的时刻my_cls这个名字被指向了counter(my_cls), 即wrapper函数。

可以打印看看print my_cls.__name__ #显示wrapper

这也是为什么能直接使用my_cls()的缘故原由,由于它已经不再是原来的类,而是新的函数wrapper。

因此需要将my_cls.STATIC_MEN修改为self.STATIC_MEN,究竟执行的时刻my_cls已经不再是原来的my_cls了

要是还想通过my_cls接见静态属性,实验以下方式

def counter(cls):

obj_list = []

@functools.wraps(cls)

def wrapper(*args, **kwargs):

... ...

return wrapper

对wrapper使用functools进行了一次包裹更新,使经由装饰的my_cls看起来更像装饰之前的类或者函数。

该历程的主要原理就是将被装饰类或者函数的部门属性直接赋值到装饰之后的工具。

如WRAPPER_ASSIGNMENTS(name, module and doc, )和WRAPPER_UPDATES(dict)等。

然则该历程不会改变wrapper是函数这样一个事实。

my_cls.__name__ == 'my_cls' and type(my_cls) is types.FunctionType

原文链接:https://www.cnblogs.com/friedCoder/p/12697364.html

本站声明:网站内容来源于网络,若有侵权,请联系我们,我们将实时处理。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值