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 logging类库使用例子
Nov 22 Python
python使用sorted函数对列表进行排序的方法
Apr 04 Python
Python的Flask框架中的Jinja2模板引擎学习教程
Jun 30 Python
python检测空间储存剩余大小和指定文件夹内存占用的实例
Jun 11 Python
python将txt文件读入为np.array的方法
Oct 30 Python
Python使用sax模块解析XML文件示例
Apr 04 Python
100行Python代码实现每天不同时间段定时给女友发消息
Sep 27 Python
关于Python中定制类的比较运算实例
Dec 19 Python
解决python-docx打包之后找不到default.docx的问题
Feb 13 Python
Python 如何调试程序崩溃错误
Aug 03 Python
Python实现学生管理系统(面向对象版)
Jun 24 Python
Python字符串的转义字符
Apr 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
php判断邮箱地址是否存在的方法
2016/02/13 PHP
利用php-cli和任务计划实现刷新token功能的方法
2017/05/03 PHP
PHP基于cookie实现统计在线人数功能示例
2019/01/16 PHP
Prototype使用指南之hash.js
2007/01/10 Javascript
JavaScript中Math对象使用说明
2008/01/16 Javascript
Js base64 加密解密介绍
2013/10/11 Javascript
Jquery注册事件实现方法
2015/05/18 Javascript
Bootstrap每天必学之基础排版
2015/11/20 Javascript
Google 地图控件集详解及实例代码
2016/08/06 Javascript
学习JavaScript图片预加载模块
2016/11/07 Javascript
jQuery Validate格式验证功能实例代码(包括重名验证)
2017/07/18 jQuery
jquery在vue脚手架中的使用方式示例
2017/08/29 jQuery
解决Vue编译时写在style中的路径问题
2017/09/21 Javascript
vue页面跳转后返回原页面初始位置方法
2018/02/11 Javascript
JavaScript封闭函数及常用内置对象示例
2019/05/13 Javascript
[02:56]DOTA2亚洲邀请赛 VG出场战队巡礼
2015/02/07 DOTA
[01:06:39]DOTA2上海特级锦标赛主赛事日 - 1 胜者组第一轮#1Liquid VS Alliance第三局
2016/03/02 DOTA
[00:16]热血竞技场
2019/03/06 DOTA
Python实现的一个简单LRU cache
2014/09/26 Python
Python操作CouchDB数据库简单示例
2015/03/10 Python
Python自定义函数定义,参数,调用代码解析
2017/12/27 Python
书单|人生苦短,你还不用python!
2017/12/29 Python
spark: RDD与DataFrame之间的相互转换方法
2018/06/07 Python
python保存文件方法小结
2018/07/27 Python
Tensorflow加载预训练模型和保存模型的实例
2018/07/27 Python
OpenCV图像颜色反转算法详解
2019/05/13 Python
使用Python制作一个打字训练小工具
2019/10/01 Python
Python3 实现爬取网站下所有URL方式
2020/01/16 Python
Pandas之read_csv()读取文件跳过报错行的解决
2020/04/21 Python
浅谈cookie和localStorage那些事
2019/08/27 HTML / CSS
HTML5页面直接调用百度地图API获取当前位置直接导航目的地的实现代码
2018/03/02 HTML / CSS
迟到检讨书2000字(精选篇)
2014/10/07 职场文书
领导工作表现评语
2015/01/04 职场文书
《蚂蚁和蝈蝈》教学反思
2016/02/22 职场文书
python中数组和列表的简单实例
2022/03/25 Python
SQLServer权限之只开启创建表权限
2022/04/12 SQL Server