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之从格式化表达式到方法
Sep 28 Python
Python脚本实现自动发带图的微博
Apr 27 Python
Python及PyCharm下载与安装教程
Nov 18 Python
Python random模块用法解析及简单示例
Dec 18 Python
Python 读写文件的操作代码
Sep 20 Python
Python中几种属性访问的区别与用法详解
Oct 10 Python
python实现简单登陆系统
Oct 18 Python
WIn10+Anaconda环境下安装PyTorch(避坑指南)
Jan 30 Python
python可视化爬虫界面之天气查询
Jul 03 Python
Python使用Slider组件实现调整曲线参数功能示例
Sep 06 Python
Python动态导入模块:__import__、importlib、动态导入的使用场景实例分析
Mar 30 Python
Python实现GIF图倒放
Jul 16 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类中private属性继承问题分析
2012/11/01 PHP
浅析php中三个等号(===)和两个等号(==)的区别
2013/08/06 PHP
php对关联数组循环遍历的实现方法
2015/03/13 PHP
phpcms手机内容页面添加上一篇和下一篇
2015/06/05 PHP
PHP+shell实现多线程的方法
2015/07/01 PHP
漂亮的thinkphp 跳转页封装示例
2019/10/16 PHP
判断用户的在线状态 onbeforeunload事件
2011/03/05 Javascript
不同Jquery版本引发的问题解决
2013/10/14 Javascript
浅谈js对象的创建和对6种继承模式的理解和遐想
2016/10/16 Javascript
详解AngularJS 模块化
2017/06/14 Javascript
JavaScript实现三级联动菜单实例代码
2017/06/26 Javascript
vue-cli之router基本使用方法详解
2017/10/17 Javascript
angularjs获取到My97DatePicker选中的值方法
2018/10/02 Javascript
微信小程序自定义弹窗wcPop插件
2018/11/19 Javascript
js canvas实现写字动画效果
2018/11/30 Javascript
bootstrap tooltips在 angularJS中的使用方法
2019/04/10 Javascript
[01:56]林书豪DOTA2上海特级锦标赛励志短片
2016/03/05 DOTA
[01:32]寻找你心中的那团火 DOTA2 TI9火焰传递活动今日开启
2019/05/16 DOTA
Python实现网站注册验证码生成类
2017/06/08 Python
python执行CMD指令,并获取返回的方法
2018/12/19 Python
Python中的pathlib.Path为什么不继承str详解
2019/06/23 Python
CSS3制作轮播图的一种方法
2019/11/11 HTML / CSS
HTML5中canvas中的beginPath()和closePath()的重要性
2018/08/24 HTML / CSS
Html5页面上如何禁止手机虚拟键盘弹出
2020/03/19 HTML / CSS
美国在线面料商店:Online Fabric Store
2018/07/26 全球购物
英国羊皮鞋类领先品牌:Just Sheepskin
2019/12/12 全球购物
若通过ObjectOutputStream向一个文件中多次以追加方式写入object,为什么用ObjectInputStream读取这些object时会产生StreamCorruptedException?
2016/10/17 面试题
生物制药毕业生自荐信
2013/10/16 职场文书
单位授权委托书范本
2014/09/26 职场文书
成都人事代理协议书
2014/10/25 职场文书
求职简历自我评价范文
2015/03/10 职场文书
今日说法观后感
2015/06/08 职场文书
《弟子规》读后感:知廉耻、明是非、懂荣辱、辨善恶
2019/12/03 职场文书
go:垃圾回收GC触发条件详解
2021/04/24 Golang
面试必问:圣杯布局和双飞翼布局的区别
2021/05/13 HTML / CSS
Python数据类型最全知识总结
2021/05/31 Python