python 踩坑系列之迟绑定

本文通过具体的Python代码示例,解释了闭包的概念及如何理解Python中的迟绑定现象,并对比了几种不同情况下闭包行为的差异。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

1.绑定

    将函数体和函数调用关联起来,就叫绑定

2.迟绑定

    在程序运行之前(也就是编译和链接时)执行的绑定是早绑定,迟绑定(late binding)是发生在运行时。

3.实例说明

def outer():
    return [lambda x: x*i for i in range(3)]
f1, f2, f3 = outer()
print f1(1)
print f2(1)
print f3(1)

>>> 2
>>> 2
>>> 2

咋一看,上面输出的值貌似和自己想的并不一致,为什么会出现这个问题,这就是python的迟绑定导致的,闭包中的i值只有在调用匿名函数的时候才会去查询,这就导致了当你在调用f1,f2,f3的时候此时i的值已变成2,所以它们的输出结果都是一样的。

我们再来看看这个问题的变形。

#情况一
l = [lambda x: x*i for i in range(3)]
f1, f2, f3 = l
print f1(1),f2(1),f3(1)

#情况二
l = [lambda x: x*i for i in range(3)]
for f in l:
    print f(1)
 
#情况三
l = (lambda x: x*i for i in range(3))
f1, f2, f3 = l
print f1(1),f2(1),f3(1)

#情况四
l = (lambda x: x*i for i in range(3))
for f in l:
    print f(1)

情况一:上面这四种情况的输出如下:

2 2 2

情况二:
2
2
2

情况三:
2 2 2

情况四:
0
1
2

 

对于情况一和情况二,没什么好说的,上面已经说过了,原因在于python的迟绑定。

但是,为什么会有情况三和情况四这种情形发生呢?

首先我们应该知道python生成器的表达式,类似于列表推导,只不过是圆括号,

(lambda x: x*i for i in range(3))其实是一个生成器,我们先说一下为什么情况四和上面三种都不一样,生成器只有在调用next()时才会执行下去,所以在对这个生成器进行for迭代时,迭代一次,i的值变化一次,所以输出结果才会有0,1,2的变化。

为什么情况三没有这个变化呢?

l = (lambda x: x*i for i in range(3)),f1, f2, f3 = l,在对f1,f2,f3进行赋值时,已全部迭代完了,此时i的值为2,所以会出现输出值全是2的情况。

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值