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中字典的基本知识初步介绍
May 21 Python
Python读大数据txt
Mar 28 Python
视觉直观感受若干常用排序算法
Apr 13 Python
Python3中的列表,元组,字典,字符串相关知识小结
Nov 10 Python
详解pandas安装若干异常及解决方案总结
Jan 10 Python
Python 把序列转换为元组的函数tuple方法
Jun 27 Python
Pytorch基本变量类型FloatTensor与Variable用法
Jan 08 Python
python GUI框架pyqt5 对图片进行流式布局的方法(瀑布流flowlayout)
Mar 12 Python
Python3 ffmpeg视频转换工具使用方法解析
Aug 10 Python
python3爬虫GIL修改多线程实例讲解
Nov 24 Python
Python+unittest+requests+excel实现接口自动化测试框架
Dec 23 Python
Python .py生成.pyd文件并打包.exe 的注意事项说明
Mar 04 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
codeigniter集成ucenter1.6双向通信的解决办法
2014/06/12 PHP
php实现文件管理与基础功能操作
2017/03/21 PHP
php使用imagecopymerge()函数创建半透明水印
2018/01/25 PHP
PHP基于curl模拟post提交json数据示例
2018/06/22 PHP
Git命令之分支详解
2021/03/02 PHP
jquery 分页控件实现代码
2009/11/30 Javascript
jQuery基础知识filter()和find()实例说明
2010/07/06 Javascript
js中的如何定位固定层的位置
2014/06/15 Javascript
三种取消选中单选框radio的方法
2014/09/09 Javascript
Javascript writable特性介绍
2015/02/27 Javascript
js插件设置innerHTML时在IE8下提示“未知运行时错误”解决方法
2015/04/25 Javascript
jquery动感漂浮导航菜单代码分享
2020/04/15 Javascript
Angularjs material 实现搜索框功能
2016/03/08 Javascript
domReady的实现案例
2016/11/23 Javascript
微信小程序自动客服功能
2017/11/02 Javascript
js 图片转base64的方式(两种)
2018/04/24 Javascript
JavaScript使用ul中li标签实现删除效果
2019/04/15 Javascript
配置node服务器并且链接微信公众号接口配置步骤详解
2019/06/21 Javascript
vue vantUI实现文件(图片、文档、视频、音频)上传(多文件)
2019/10/15 Javascript
python关键字and和or用法实例
2015/05/28 Python
python实现文件快照加密保护的方法
2015/06/30 Python
Python聚类算法之基本K均值实例详解
2015/11/20 Python
Python返回数组/List长度的实例
2018/06/23 Python
Python实现FTP弱口令扫描器的方法示例
2019/01/31 Python
实例讲解Python3中abs()函数
2019/02/19 Python
Python+OpenCV实现实时眼动追踪的示例代码
2019/11/11 Python
Python filter()及reduce()函数使用方法解析
2020/09/05 Python
HTML5 文件域+FileReader 分段读取文件并上传到服务器
2017/10/23 HTML / CSS
英国标志性奢侈品牌:Burberry
2016/07/28 全球购物
学前班评语大全
2014/05/04 职场文书
人力资源求职信
2014/05/25 职场文书
欢迎领导标语
2014/06/27 职场文书
单位委托书怎么写
2014/08/02 职场文书
租房协议书范文
2014/08/20 职场文书
安全生产先进个人总结
2015/02/15 职场文书
投诉书格式范本
2015/07/02 职场文书