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实现360的字符显示界面
Feb 21 Python
Python实现过滤单个Android程序日志脚本分享
Jan 16 Python
用Python编写一个简单的俄罗斯方块游戏的教程
Apr 03 Python
使用Python的PEAK来适配协议的教程
Apr 14 Python
python操作mongodb根据_id查询数据的实现方法
May 20 Python
python 捕获 shell/bash 脚本的输出结果实例
Jan 04 Python
Django框架自定义模型管理器与元选项用法分析
Jul 22 Python
详解Python中namedtuple的使用
Apr 27 Python
Python中and和or如何使用
May 28 Python
使用pyplot.matshow()函数添加绘图标题
Jun 16 Python
Python爬虫进阶之Beautiful Soup库详解
Apr 29 Python
Python用tkinter实现自定义记事本的方法详解
Mar 31 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
为查询结果建立向后/向前按钮
2006/10/09 PHP
php 邮件发送问题解决
2014/03/22 PHP
php的SimpleXML方法读写XML接口文件实例解析
2014/06/16 PHP
jQuery第三课 修改元素属性及内容的代码
2010/03/14 Javascript
Jquery数独游戏解析(一)-页面布局
2010/11/05 Javascript
jQuery中调用WebService方法小结
2011/03/28 Javascript
node.js中的buffer.Buffer.isEncoding方法使用说明
2014/12/14 Javascript
简单谈谈javascript代码复用模式
2015/01/28 Javascript
Javascript核心读书有感之词法结构
2015/02/01 Javascript
JQuery实现可直接编辑的表格
2015/04/16 Javascript
简述JavaScript中正则表达式的使用方法
2015/06/15 Javascript
原生JS实现仿淘宝网左侧商品分类菜单效果代码
2015/09/10 Javascript
js实现的彩色方块飞舞奇幻效果
2016/01/27 Javascript
AngularJS实现Input格式化的方法
2016/11/07 Javascript
jQuery Validate插件自定义验证规则的方法
2016/12/27 Javascript
AngularJS1.X学习笔记2-数据绑定详解
2017/04/01 Javascript
详解vue mint-ui源码解析之loadmore组件
2017/10/11 Javascript
Angular 4.x+Ionic3踩坑之Ionic 3.x界面传值详解
2018/03/13 Javascript
jQuery实现轮播图及其原理详解
2020/04/12 jQuery
jqGrid表格底部汇总、合计行footerrow处理
2019/08/21 Javascript
Vue.js组件props数据验证实现详解
2019/10/19 Javascript
关于vue利用postcss-pxtorem进行移动端适配的问题
2019/11/20 Javascript
JS实现点星星消除小游戏
2020/03/24 Javascript
vue跳转页面的几种方法(推荐)
2020/03/26 Javascript
sharp.js安装过程中遇到的问题总结
2020/04/02 Javascript
vue 使用 sortable 实现 el-table 拖拽排序功能
2020/12/26 Vue.js
python框架中flask知识点总结
2018/08/17 Python
Windows系统下PhantomJS的安装和基本用法
2018/10/21 Python
Django JWT Token RestfulAPI用户认证详解
2019/01/23 Python
如何运行.ipynb文件的图文讲解
2019/06/27 Python
python flask中动态URL规则详解
2019/11/22 Python
简述你对Statement,PreparedStatement,CallableStatement的理解
2013/03/25 面试题
银行党员批评与自我批评
2014/10/15 职场文书
pytorch损失反向传播后梯度为none的问题
2021/05/12 Python
教你使用TensorFlow2识别验证码
2021/06/11 Python
CSS基础详解
2021/10/16 HTML / CSS