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实现将不规范的英文名字首字母大写
Nov 15 Python
Python实现的密码强度检测器示例
Aug 23 Python
python字典的遍历3种方法详解
Aug 10 Python
PYTHON EVAL的用法及注意事项解析
Sep 06 Python
python 单线程和异步协程工作方式解析
Sep 28 Python
AUC计算方法与Python实现代码
Feb 28 Python
Python 去除字符串中指定字符串
Mar 05 Python
解决pyqt5异常退出无提示信息的问题
Apr 08 Python
python数据分析工具之 matplotlib详解
Apr 09 Python
pytorch VGG11识别cifar10数据集(训练+预测单张输入图片操作)
Jun 24 Python
Python之matplotlib绘制饼图
Apr 13 Python
彻底弄懂Python中的回调函数(callback)
Jun 25 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读取MySQL数据代码
2008/06/05 PHP
为你总结一些php信息函数
2015/10/21 PHP
php两点地理坐标距离的计算方法
2018/12/29 PHP
JavaScript实现动态增加文件域表单
2009/02/12 Javascript
jQuery实现仿淘宝带有指示条的图片转动切换效果完整实例
2015/03/04 Javascript
浅谈addEventListener和attachEvent的区别
2016/07/14 Javascript
EasyUI学习之Combobox级联下拉列表(2)
2016/12/29 Javascript
Node.js  REPL (交互式解释器)实例详解
2017/08/06 Javascript
详解.vue文件中style标签的几个标识符
2018/07/17 Javascript
vue Tab切换以及缓存页面处理的几种方式
2019/11/05 Javascript
vue项目中在可编辑div光标位置插入内容的实现代码
2020/01/07 Javascript
[04:04]DOTA2亚洲邀请赛比赛场馆&酒店全攻略
2017/03/23 DOTA
介绍Python的Django框架中的静态资源管理器django-pipeline
2015/04/25 Python
解读Python中degrees()方法的使用
2015/05/18 Python
Python脚本简单实现打开默认浏览器登录人人和打开QQ的方法
2016/04/12 Python
python for循环输入一个矩阵的实例
2018/11/14 Python
Python验证码截取识别代码实例
2020/05/16 Python
基于python检查矩阵计算结果
2020/05/21 Python
Python 如何测试文件是否存在
2020/07/31 Python
python按照list中字典的某key去重的示例代码
2020/10/13 Python
Python类class参数self原理解析
2020/11/19 Python
CSS3绘制六边形的简单实现
2016/08/25 HTML / CSS
国外最大的眼镜网站:Coastal
2017/08/09 全球购物
家乐福台湾线上购物网:Carrefour台湾
2020/09/15 全球购物
数学专业推荐信范文
2013/11/21 职场文书
电子邮箱格式怎么写
2014/01/12 职场文书
志愿者服务感言
2014/02/27 职场文书
学党史心得体会
2014/09/05 职场文书
婚宴邀请函
2015/01/30 职场文书
巴黎圣母院观后感
2015/06/10 职场文书
2015大学迎新晚会策划书
2015/07/16 职场文书
讲座新闻稿
2015/07/18 职场文书
导游词之苏州寒山寺
2019/12/05 职场文书
Nginx如何配置Http、Https、WS、WSS的方法步骤
2021/05/11 Servers
Redis集群新增、删除节点以及动态增加内存的方法
2021/09/04 Redis
C#连接ORACLE出现乱码问题的解决方法
2021/10/05 Oracle