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 22 Python
python 实现上传图片并预览的3种方法(推荐)
Jul 14 Python
python的socket编程入门
Jan 29 Python
Python按钮的响应事件详解
Mar 04 Python
一文秒懂python读写csv xml json文件各种骚操作
Jul 04 Python
解决Django加载静态资源失败的问题
Jul 28 Python
PyTorch预训练的实现
Sep 18 Python
python tkinter canvas使用实例
Nov 04 Python
python双向链表原理与实现方法详解
Dec 03 Python
python GUI库图形界面开发之PyQt5窗口控件QWidget详细使用方法
Feb 26 Python
浅谈pytorch 模型 .pt, .pth, .pkl的区别及模型保存方式
May 25 Python
如何利用Python实现一个论文降重工具
Jul 09 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
德生PL450的电路分析和低放电路的改进办法
2021/03/02 无线电
冰滴咖啡制作步骤
2021/03/03 冲泡冲煮
Confirmer JQuery确认对话框组件
2010/06/09 Javascript
JQuery实现倒计时按钮的实现代码
2012/03/23 Javascript
jQuery UI Autocomplete 1.8.16 中文输入修正代码
2012/04/16 Javascript
Javascript函数的参数
2015/07/16 Javascript
js实现黑色简易的滑动门网页tab选项卡效果
2015/08/31 Javascript
JS基于面向对象实现的拖拽库实例
2015/09/24 Javascript
利用CSS3在Angular中实现动画
2016/01/15 Javascript
javascript实现PC网页里的拖拽效果
2016/03/14 Javascript
JavaScript驾驭网页-获取网页元素
2016/03/24 Javascript
jQuery实现图像旋转动画效果
2016/05/29 Javascript
JS弹出窗口的运用与技巧大全
2016/11/01 Javascript
jQuery插件FusionWidgets实现的Bulb图效果示例【附demo源码下载】
2017/03/23 jQuery
基于JavaScript实现五子棋游戏
2020/08/26 Javascript
react实现复选框全选和反选组件效果
2020/08/25 Javascript
[37:47]IG vs Winstrike 2018国际邀请赛小组赛BO2 第二场 8.19
2018/08/21 DOTA
[01:54]TI珍贵瞬间系列(三):翻盘
2020/08/28 DOTA
python ip正则式
2009/05/07 Python
python将html转成PDF的实现代码(包含中文)
2013/03/04 Python
用python登录Dr.com思路以及代码分享
2014/06/25 Python
python对字典进行排序实例
2014/09/25 Python
python中as用法实例分析
2015/04/30 Python
Python3 中文文件读写方法
2018/01/23 Python
编写多线程Python服务器 最适合基础
2018/09/14 Python
python实现将文件夹下面的不是以py文件结尾的文件都过滤掉的方法
2018/10/21 Python
使用python的pyplot绘制函数实例
2020/02/13 Python
Vichy薇姿加拿大官网:法国药妆,全球专业敏感肌护肤领先品牌
2018/07/11 全球购物
英国领先的在线高尔夫商店:Gamola Golf
2019/11/16 全球购物
应用电子技术专业个人求职信
2013/09/21 职场文书
航空学院求职信
2014/06/11 职场文书
胡雪岩故居导游词
2015/02/06 职场文书
迎新年主持词
2015/07/06 职场文书
2016入党培训心得体会范文
2016/01/08 职场文书
祝福语集锦:给妹妹结婚的祝福语
2019/12/18 职场文书
Python实现信息管理系统
2022/06/05 Python