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

作者:mrr 时间:2023-11-29 14:01:09 

什么是闭包?

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

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

延迟绑定

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

Python


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


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


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


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,闭包
0
投稿

猜你喜欢

  • 详解Python学习之安装pandas

    2021-04-30 19:35:33
  • 高亮闪烁某个元素的js脚本

    2024-04-16 09:04:27
  • Kibo 用于处理键盘事件的Javascript工具库

    2024-04-10 10:45:53
  • golang如何实现抓取IP地址的蜘蛛程序详解

    2024-04-25 15:08:05
  • Python用正则表达式实现爬取古诗文网站信息

    2021-08-30 07:12:51
  • Microsoft SQL Server数据库各版本下载地址集合

    2024-01-28 12:15:07
  • pandas数据选取:df[] df.loc[] df.iloc[] df.ix[] df.at[] df.iat[]

    2023-04-19 20:18:21
  • 玩转MySQL中的外键约束之PHP篇

    2010-03-18 10:20:00
  • mysql 5.7.14 安装配置简单教程

    2024-01-13 04:41:48
  • PyTorch梯度裁剪避免训练loss nan的操作

    2022-02-16 10:56:23
  • sql存储过程的使用和介绍

    2024-01-14 01:58:19
  • JavaScript开发时的五个小提示

    2007-11-21 19:54:00
  • Mysql CAST函数的具体使用

    2024-01-16 16:17:04
  • IE不支持overrideMimeType()方法,即使是IE7.

    2009-02-08 16:58:00
  • [欣赏] 情景互动广告

    2008-08-06 12:59:00
  • 使用numpy.mean() 计算矩阵均值方式

    2021-12-17 03:55:53
  • Javascript连接Access数据库完整实例

    2024-01-15 22:16:58
  • python实现简单贪吃蛇小游戏

    2021-12-11 13:59:00
  • Python使用微信接入图灵机器人过程解析

    2022-05-29 04:03:14
  • 如何调用Oracle存储过程?

    2009-11-15 20:13:00
  • asp之家 网络编程 m.aspxhome.com