Python延迟绑定问题原理及解决方案


Posted in Python onAugust 04, 2020

延迟绑定出现在闭包问题中。下面我们看一个闭包的例子:

def (n):
  def mul(x):
    return n*x
  return mul
double = gen_mul(2)
doubled_value = double(6)

可以看出满足闭包的几点:

  • 有内部函数
  • 内部函数引用了外部函数中的自由变量
  • 内部函数被返回

闭包的优点:

  • 可以避免使用全局变量
  • 可以持久化变量,达到静态变量的作用

闭包的缺点:

  • 可能会消耗大量的内存
  • 可能会导致内存泄漏

当然缺点可以通过人为避免。

现在我们来看看另一个会引出延迟绑定的例子:

def multipliers():
  return [lambda x : i * x for i in range(4)]
print([m(2) for m in multipliers()]) # [6,6,6,6]

上边的例子会输出[6,6,6,6],而不是我们预期的[0,2,4,6]。

这就是延迟绑定导致的结果。具体过程我们可以来分析下:
执行第三行时,会先执行multipliers函数,然后执行函数中的列表解析式。在每一次迭代的时候都会生成一个匿名函数(这里只是定义)作为元素。然后回到第三行,遍历返回的列表中的匿名函数,传入参数2并执行。此时函数类似于这样:

def noname(x):
return i * x

我们知道Python查找变量的作用域链的顺序依次为LEGB:

局部变量(L)->外部函数中的局部变量(E)->全局变量(G)->内置变量(B)

非常重要的一点我们需要知道:Python的作用域在编译时就已经形成了,而不是在运行时,函数的作用域与其被调用的位置无关。

那么在本例中,上面的noname函数体中的i从何而来呢?当然首先会到multipliers函数的局部变量中去寻找。此时i的值已经为3,所以出现这种让人”费解”的现象。

那么现在我们既然已经知道了原因,那么要怎样解决呢?

我们可以将迭代的i值直接注入到匿名函数的函数体中,这里给出两种方法:

通过为参数设置默认值,这是因为在编译时就会计算确定默认值:

def multipliers_ch1():
return [lambda m,x=i : m * x for i in range(4)]

通过内置函数partial:

from functools import partial
def multipliers_ch2():
  return [partial(lambda m,x : m * x,i) for i in range(4)]

利用生成器的延迟计算:

def multipliers_ch3():
  for m in range(4):
    yield lambda x: m * x

partial及生成器的内容会在以后分享。

运行结果

print([m(2) for m in multipliers_ch1()]) # [0,2,4,6]
print([m(2) for m in multipliers_ch2()]) # [0,2,4,6]
print([m(2) for m in multipliers_ch3()]) # [0,2,4,6]

注:

自由变量:指未在本地作用域中绑定的变量,我们可通过访问函数的code属性进行查看:

fun.code.co_freevars

LEGB: 可看该部分解释

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Python 相关文章推荐
Python连接MySQL并使用fetchall()方法过滤特殊字符
Mar 13 Python
Python中使用插入排序算法的简单分析与代码示例
May 04 Python
Zookeeper接口kazoo实例解析
Jan 22 Python
python利用微信公众号实现报警功能
Jun 10 Python
Python递归函数实例讲解
Feb 27 Python
关于Numpy数据类型对象(dtype)使用详解
Nov 27 Python
安装多个版本的TensorFlow的方法步骤
Apr 21 Python
python利用Excel读取和存储测试数据完成接口自动化教程
Apr 30 Python
浅谈Pycharm的项目文件名是红色的原因及解决方式
Jun 01 Python
OpenCV4.1.0+VS2017环境配置的方法步骤
Jul 09 Python
浅谈盘点5种基于Python生成的个性化语音方法
Feb 05 Python
python3中apply函数和lambda函数的使用详解
Feb 28 Python
Python 使用生成器代替线程的方法
Aug 04 #Python
详解Tensorflow不同版本要求与CUDA及CUDNN版本对应关系
Aug 04 #Python
python读取xml文件方法解析
Aug 04 #Python
如何利用python进行时间序列分析
Aug 04 #Python
通过实例简单了解Python sys.argv[]使用方法
Aug 04 #Python
哪种Python框架适合你?简单介绍几种主流Python框架
Aug 04 #Python
python logging 重复写日志问题解决办法详解
Aug 04 #Python
You might like
PHP处理JSON字符串key缺少双引号的解决方法
2014/09/16 PHP
jQuery EasyUI API 中文文档 - TimeSpinner时间微调器
2011/10/23 Javascript
jquery的live使用注意事项
2014/02/18 Javascript
JS简单操作select和dropdownlist实例
2014/11/26 Javascript
js鼠标滑过图片震动特效的方法
2015/02/17 Javascript
javascript通过获取html标签属性class实现多选项卡的方法
2015/07/27 Javascript
jquery.cookie实现的客户端购物车操作实例
2015/12/24 Javascript
一波JavaScript日期判断脚本分享
2016/03/06 Javascript
jquery操作checkbox火狐下第二次无法勾选的解决方法
2016/10/10 Javascript
利用CSS、JavaScript及Ajax实现图片预加载的方法
2016/11/29 Javascript
Flask中获取小程序Request数据的两种方法
2017/05/12 Javascript
彻底弄懂 JavaScript 执行机制
2018/10/23 Javascript
微信小程序 腾讯地图SDK 获取当前地址实现解析
2019/08/12 Javascript
JS实现可用滑块滑动的缓动图代码
2019/09/01 Javascript
微信小程序开发(三):返回上一级页面并刷新操作示例【页面栈】
2020/06/01 Javascript
[01:34]2016国际邀请赛中国区预选赛IG战队教练采访
2016/06/27 DOTA
[01:20:38]完美世界DOTA2联赛 GXR vs IO 第一场 11.07
2020/11/09 DOTA
python实现的各种排序算法代码
2013/03/04 Python
举例讲解Python面向对象编程中类的继承
2016/06/17 Python
Python 数据结构之队列的实现
2017/01/22 Python
python使用matplotlib绘图时图例显示问题的解决
2017/04/27 Python
Python正则表达式经典入门教程
2017/05/22 Python
Python使用Scrapy保存控制台信息到文本解析
2017/12/27 Python
python实现微信发送邮件关闭电脑功能
2018/02/22 Python
numpy求平均值的维度设定的例子
2019/08/24 Python
解决torch.autograd.backward中的参数问题
2020/01/07 Python
Python Socketserver实现FTP文件上传下载代码实例
2020/03/27 Python
Jupyter notebook如何实现指定浏览器打开
2020/05/13 Python
PyCharm2020最新激活码+激活码补丁(亲测最新版PyCharm2020.2激活成功)
2020/11/25 Python
css3实现一个div设置多张背景图片及background-image属性实例演示
2017/08/10 HTML / CSS
彪马土耳其官网:PUMA土耳其
2019/07/14 全球购物
对教师的评语
2014/04/28 职场文书
2015元旦联欢晚会结束语
2014/12/14 职场文书
电影焦裕禄观后感
2015/06/09 职场文书
2015年信息化建设工作总结
2015/07/23 职场文书
Mysql服务添加 iptables防火墙策略的方案
2021/04/29 MySQL