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下如何让web元素的生成更简单的分析
Jul 17 Python
Python闭包的两个注意事项(推荐)
Mar 20 Python
python通过百度地图API获取某地址的经纬度详解
Jan 28 Python
利用Python实现在同一网络中的本地文件共享方法
Jun 04 Python
python+flask实现API的方法
Nov 21 Python
python绘制漏斗图步骤详解
Mar 04 Python
使用python的pandas为你的股票绘制趋势图
Jun 26 Python
python wav模块获取采样率 采样点声道量化位数(实例代码)
Jan 22 Python
python 多线程共享全局变量的优劣
Sep 24 Python
python 解决Windows平台上路径有空格的问题
Nov 10 Python
python字典按照value排序方法
Dec 28 Python
python中if和elif的区别介绍
Nov 07 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
动漫女神老婆无限好,但日本女生可能就不是这么一回事了!
2020/03/04 日漫
Adodb的十个实例(清晰版)
2006/12/31 PHP
超强分页类2.0发布,支持自定义风格,默认4种显示模式
2007/01/02 PHP
PHP5中虚函数的实现方法分享
2011/04/20 PHP
php字符串分割函数explode的实例代码
2013/02/07 PHP
php天翼开放平台短信发送接口实现方法
2014/12/22 PHP
PHP + plupload.js实现多图上传并显示进度条加删除实例代码
2017/03/06 PHP
JQuery从头学起第三讲
2010/07/06 Javascript
juqery 学习之三 选择器 子元素与表单
2010/11/25 Javascript
JQuery动态给table添加、删除行 改进版
2011/01/19 Javascript
JavaScript中把数字转换为字符串的程序代码
2013/06/19 Javascript
使用js判断控件是否获得焦点
2014/01/03 Javascript
jquery带翻页动画的电子杂志代码分享
2015/08/21 Javascript
JavaScript判断表单为空及获取焦点的方法
2016/02/12 Javascript
JS Array创建及concat()split()slice()的使用方法
2016/06/03 Javascript
如何使用Vuex+Vue.js构建单页应用
2016/10/27 Javascript
原生javascript实现图片放大镜效果
2017/01/18 Javascript
jQuery返回定位插件详解
2017/05/15 jQuery
Vue.js实现备忘录功能
2019/06/26 Javascript
15个简单的JS编码标准让你的代码更整洁(小结)
2020/07/16 Javascript
python二叉树遍历的实现方法
2013/11/21 Python
Python基于pygame实现的弹力球效果(附源码)
2015/11/11 Python
Python中使用装饰器来优化尾递归的示例
2016/06/18 Python
python实现应用程序在右键菜单中添加打开方式功能
2017/01/09 Python
使用pycharm设置控制台不换行的操作方法
2019/01/19 Python
pycharm创建一个python包方法图解
2019/04/10 Python
django框架model orM使用字典作为参数,保存数据的方法分析
2019/06/24 Python
Python解决pip install时出现的Could not fetch URL问题
2019/08/01 Python
给Python学习者的文件读写指南(含基础与进阶)
2020/01/29 Python
使用Filters滤镜弥补CSS3的跨浏览器问题以及兼容低版本IE
2013/01/23 HTML / CSS
英国家喻户晓的折扣商场:TK Maxx
2017/05/26 全球购物
NET程序员上机面试题
2015/05/23 面试题
P/Invoke是什么
2015/07/31 面试题
大学生的网络创业计划书
2013/12/26 职场文书
新员工试用期自我鉴定
2014/04/17 职场文书
春秋淹城导游词
2015/02/11 职场文书