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中的编码知识整理汇总
Jan 26 Python
python 编程之twisted详解及简单实例
Jan 28 Python
python中defaultdict的用法详解
Jun 07 Python
Python3爬虫学习之爬虫利器Beautiful Soup用法分析
Dec 12 Python
django模板结构优化的方法
Feb 28 Python
Django 通过JS实现ajax过程详解
Jul 30 Python
给你一面国旗 教你用python画中国国旗
Sep 24 Python
Python列表原理与用法详解【创建、元素增加、删除、访问、计数、切片、遍历等】
Oct 30 Python
python cv2截取不规则区域图片实例
Dec 21 Python
Python多线程操作之互斥锁、递归锁、信号量、事件实例详解
Mar 24 Python
Anaconda的安装与虚拟环境建立
Nov 18 Python
python 实现一个简单的线性回归案例
Dec 17 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中替换换行符的几种方法小结
2012/10/15 PHP
php中get_headers函数的作用及用法的详细介绍
2013/04/27 PHP
如何批量清理系统临时文件(语言:C#、 C/C++、 php 、python 、java )
2016/02/01 PHP
PHP编写学校网站上新生注册登陆程序的实例分享
2016/03/21 PHP
jquery tablesorter.js 支持中文表格排序改进
2009/12/09 Javascript
Jquery 获取checkbox的checked问题
2011/11/16 Javascript
事件冒泡是什么如何用jquery阻止事件冒泡
2013/03/20 Javascript
jQuery实现点击图片翻页展示效果的方法
2015/02/16 Javascript
JavaScript实现拖拽网页内元素的方法
2015/04/15 Javascript
iframe跨域通信封装详解
2015/08/11 Javascript
JavaScript 实现的 zip 压缩和解压缩工具包Zip.js使用详解
2015/12/14 Javascript
js添加绑定事件的方法
2016/05/15 Javascript
Javascript blur与click冲突解决办法
2017/01/09 Javascript
用js制作淘宝放大镜效果
2020/10/28 Javascript
JS+canvas实现的五子棋游戏【人机大战版】
2017/07/19 Javascript
浅谈关于iview表单验证的问题
2018/09/29 Javascript
详解a标签添加onclick事件的几种方式
2019/03/29 Javascript
基于elementUI竖向表格、和并列的案例
2020/10/26 Javascript
[02:17]2016国际邀请赛中国区预选赛VG战队领队采访
2016/06/26 DOTA
Python中使用item()方法遍历字典的例子
2014/08/26 Python
python中的五种异常处理机制介绍
2014/09/02 Python
零基础写python爬虫之urllib2使用指南
2014/11/05 Python
Python操作MongoDB数据库PyMongo库使用方法
2015/04/27 Python
分析Python中设计模式之Decorator装饰器模式的要点
2016/03/02 Python
python出现"IndentationError: unexpected indent"错误解决办法
2017/10/15 Python
python对视频画框标记后保存的方法
2018/12/07 Python
python简单区块链模拟详解
2019/07/03 Python
使用PyTorch训练一个图像分类器实例
2020/01/08 Python
使用OpenCV校准鱼眼镜头的方法
2020/11/26 Python
万户网络JAVA程序员岗位招聘笔试试卷
2013/01/08 面试题
网游商务专员求职信
2013/10/15 职场文书
外国语学院毕业生自荐信
2013/10/28 职场文书
先进教师个人主要事迹材料
2015/11/03 职场文书
2016年法制宣传月活动总结
2016/04/01 职场文书
如何让2019年上半年的工作总结更出色!
2019/07/01 职场文书
大学生创业,为什么都会选择快餐饮?
2019/08/08 职场文书