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 相关文章推荐
wxPython 入门教程
Oct 07 Python
Python(Tornado)模拟登录小米抢手机
Nov 12 Python
python基础教程之类class定义使用方法
Feb 20 Python
python字符串排序方法
Aug 29 Python
python 读写中文json的实例详解
Oct 29 Python
python的Tqdm模块的使用
Jan 10 Python
OpenCV-Python 摄像头实时检测人脸代码实例
Apr 30 Python
Tensorflow的梯度异步更新示例
Jan 23 Python
python3实现raspberry pi(树莓派)4驱小车控制程序
Feb 12 Python
python实现ftp文件传输系统(案例分析)
Mar 20 Python
python基于爬虫+django,打造个性化API接口
Jan 21 Python
Python LMDB库的使用示例
Feb 14 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
一个用于MySQL的PHP XML类
2006/10/09 PHP
dedecms模板标签代码官方参考
2007/03/17 PHP
php利用事务处理转账问题
2015/04/22 PHP
PHP实现上一篇下一篇的方法实例总结
2016/09/22 PHP
ThinkPHP3.2框架使用addAll()批量插入数据的方法
2017/03/16 PHP
javascript语言结构小记(一)
2011/09/10 Javascript
node.js中的fs.createReadStream方法使用说明
2014/12/17 Javascript
谷歌浏览器调试JavaScript小技巧
2014/12/29 Javascript
使用AngularJS对路由进行安全性处理的方法
2015/06/18 Javascript
JS中常用的正则表达式
2016/09/29 Javascript
jQuery的Cookie封装,与PHP交互的简单实现
2016/10/05 Javascript
JavaScript实现垂直滚动条效果
2017/01/18 Javascript
ES6中Proxy代理用法实例浅析
2017/04/06 Javascript
jQuery插件DataTables分页开发心得体会
2017/08/22 jQuery
VueJs监听window.resize方法示例
2018/01/17 Javascript
JavaScript实现JSON合并操作示例【递归深度合并】
2018/09/07 Javascript
vue+web端仿微信网页版聊天室功能
2019/04/30 Javascript
微信小程序实现菜单左右联动
2020/05/19 Javascript
JavaScript 几种循环方式以及模块化的总结
2020/09/03 Javascript
[11:57]《一刀刀一天》第十七期:TI中国军团加油!
2014/05/26 DOTA
python爬取足球直播吧五大联赛积分榜
2018/06/13 Python
在python image 中安装中文字体的实现方法
2019/08/22 Python
已安装tensorflow-gpu,但keras无法使用GPU加速的解决
2020/02/07 Python
Python 实现Image和Ndarray互相转换
2020/02/19 Python
python 实现socket服务端并发的四种方式
2020/12/14 Python
维也纳通行证:Vienna PASS
2019/07/18 全球购物
美国最大的购物网站:Amazon.com(亚马逊美国)
2020/05/23 全球购物
网上常见的一份Linux面试题(多项选择部分)
2015/02/07 面试题
编辑个人求职信范文
2013/09/21 职场文书
国庆节演讲稿
2014/05/27 职场文书
党的群众路线教育实践活动宣传标语口号
2014/06/06 职场文书
2015年评职称工作总结范文
2015/04/20 职场文书
2016优秀教师先进个人事迹材料
2016/02/25 职场文书
我的收音机情缘
2022/04/05 无线电
使用Nginx+Tomcat实现负载均衡的全过程
2022/05/30 Servers
Mysql的Table doesn't exist问题及解决
2022/12/24 MySQL