gpt4 book ai didi

Python闭包的两个注意事项(推荐)

转载 作者:qq735679552 更新时间:2022-09-27 22:32:09 24 4
gpt4 key购买 nike

CFSDN坚持开源创造价值,我们致力于搭建一个资源共享平台,让每一个IT人在这里找到属于你的精彩世界.

这篇CFSDN的博客文章Python闭包的两个注意事项(推荐)由作者收集整理,如果你对这篇文章有兴趣,记得点赞哟.

什么是闭包?

简单说,闭包就是根据不同的配置信息得到不同的结果.

再来看看专业的解释:闭包(Closure)是词法闭包(Lexical Closure)的简称,是引用了自由变量的函数。这个被引用的自由变量将和这个函数一同存在,即使已经离开了创造它的环境也不例外。所以,有另一种说法认为闭包是由函数和与其相关的引用环境组合而成的实体.

延迟绑定 。

Python闭包函数所引用的外部自由变量是延迟绑定的.

Python 。

?
1
2
3
4
5
6
7
8
In [ 2 ]: def multipliers():
   ...:   return [ lambda x: i * x for i in range ( 4 )]
In [ 3 ]: print [m( 2 ) for m in multipliers()]
[ 6 , 6 , 6 , 6 ]
In [ 2 ]: def multipliers():
   ...:   return [ lambda x: i * x for i in range ( 4 )]
In [ 3 ]: print [m( 2 ) for m in multipliers()]
[ 6 , 6 , 6 , 6 ]

如以上代码: i是闭包函数引用的外部作用域的自由变量, 只有在内部函数被调用的时候才会搜索变量i的值, 由于循环已结束, i指向最终值3, 所以各函数调用都得到了相同的结果.

解决方法

1) 生成闭包函数的时候立即绑定(使用函数形参的默认值)

Python 。

?
1
2
3
4
5
6
7
8
9
10
In [ 5 ]: def multipliers():
   return [ lambda x, i = i: i * x for i in range ( 4 )]
     ...:
In [ 6 ]: print [m( 2 ) for m in multipliers()]
[ 0 , 2 , 4 , 6 ]
In [ 5 ]: def multipliers():
   return [ lambda x, i = i: i * x for i in range ( 4 )]
     ...:
In [ 6 ]: print [m( 2 ) for m in multipliers()]
[ 0 , 2 , 4 , 6 ]

如以上代码: 生成闭包函数的时候, 可以看到每个闭包函数都有一个带默认值的参数: i=i, 此时, 解释器会查找i的值, 并将其赋予形参i, 这样在生成闭包函数的外部作用域(即外部循环中), 找到了变量i, 遂将其当前值赋予形参i.

2) 使用functools.partial

Python 。

?
1
2
3
4
5
6
7
8
9
10
In [ 26 ]: def multipliers():
   return [functools.partial( lambda i, x: x * i, i) for i in range ( 4 )]
   ....:
In [ 27 ]: print [m( 2 ) for m in multipliers()]
   [ 0 , 2 , 4 , 6 ]
In [ 26 ]: def multipliers():
   return [functools.partial( lambda i, x: x * i, i) for i in range ( 4 )]
   ....:
In [ 27 ]: print [m( 2 ) for m in multipliers()]
   [ 0 , 2 , 4 , 6 ]

如以上代码: 在有可能因为延迟绑定而出问题的时候, 可以通过functools.partial构造偏函数, 使得自由变量优先绑定到闭包函数上.

禁止在闭包函数内对引用的自由变量进行重新绑定 。

Python 。

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
def foo(func):
   free_value = 8
   def _wrapper( * args, * * kwargs):
     old_free_value = free_value #保存旧的free_value
     free_value = old_free_value * 2 #模拟产生新的free_value
     func( * args, * * kwargs)
     free_value = old_free_value
   return _wrapper
def foo(func):
   free_value = 8
   def _wrapper( * args, * * kwargs):
     old_free_value = free_value #保存旧的free_value
     free_value = old_free_value * 2 #模拟产生新的free_value
     func( * args, * * kwargs)
     free_value = old_free_value
   return _wrapper

以上代码会报错, UnboundLocalError: local variable 'free_value' referenced before assignment, 以上代码本意是打算实现一个带有某个初始化状态(free_value)但在执行内部闭包函数的时候又可以按需变化出新的状态(free_value = old_free_value * 2)的装饰器, 但内部由于发生了重新绑定, 解释器会将free_value看作局部变量, old_free_value = free_value则会报错, 因为解释器认为free_value是没有赋值就被引用了.

解决:

打算修改闭包函数引用的自由变量时, 可以将其放入一个list, 这样, free_value = [8], free_value不可修改, 但free_value[0]是可以安全的被修改的.

另外, Python 3.x增加了nonlocal关键字, 也可以解决这个问题.

以上所述是小编给大家介绍的Python闭包的两个注意事项,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对我网站的支持! 。

最后此篇关于Python闭包的两个注意事项(推荐)的文章就讲到这里了,如果你想了解更多关于Python闭包的两个注意事项(推荐)的内容请搜索CFSDN的文章或继续浏览相关文章,希望大家以后支持我的博客! 。

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com