时间:2021-05-22
延迟绑定出现在闭包问题中。下面我们看一个闭包的例子:
def (n): def mul(x): return n*x return muldouble = 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 partialdef 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 * xpartial及生成器的内容会在以后分享。
运行结果
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: 可看该部分解释
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。
声明:本页内容来源网络,仅供用户参考;我单位不保证亦不表示资料全面及准确无误,也不保证亦不表示这些资料为最新信息,如因任何原因,本网内容或者用户因倚赖本网内容造成任何损失或损害,我单位将不会负任何法律责任。如涉及版权问题,请提交至online#300.cn邮箱联系删除。
CSS多浏览器兼容性问题及解决方案一、document.formName.item(”itemName”)问题问题说明:IE下,可以使用document.for
这篇文章主要介绍了Python3的unicode编码转换成中文的问题及解决方案,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要
我们专业为跨境公司提供跨境国际网络专线综合解决方案,解决跨境访问出现的网络缓慢、延迟、卡、掉线、无法访问等问题,提供一条可高速访问、通畅稳定、安全的国际网络专线
在数据库或者请求操作时,如果选择的时间段过短或操作数据量过大,就会遇到"请求超时"的的问题,网络上提供很多解决方案,但普遍不完善,根据个人经验及参考网络解决方案
vue双向绑定数据如何限制长度?具体方法请阅读文章问题描述vue中输入框v-modle双向绑定的数据;在需要的业务场景下需要对其进行字数长度限制;解决方案可以使