Python中的延迟绑定

延迟绑定出现在闭包问题中。下面我们看一个闭包的例子:

def gen_mul(n):
    def mul(x):
        return n*x
    return mul

double = gen_mul(2)
double_value = double(6)

print(double_value)

  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9

可以看出满足闭包的几点:

  • 有内部函数
  • 内部函数引用了外部函数中的自由变量
  • 内部函数被返回

闭包的优点:

  • 可以避免使用全局变量
  • 可以持久化变量,达到静态变量的作用

闭包的缺点:

  • 可能会消耗大量的内存

  • 可能会导致内存泄漏

现在我们来看看另一个会引出延迟绑定的例子:

'''
遇到问题没人解答?小编创建了一个Python学习交流QQ群:7784639393
寻找有志同道合的小伙伴,互帮互助,群里还有不错的视频学习教程和PDF电子书!
'''
def multipliers():
    return [lambda x : x*i for i in range(4)]

print([m(2) for m in multipliers()])

  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8

上边的例子会输出[6,6,6,6],而不是我们预期的[0,2,4,6]。

这就是延迟绑定导致的结果。具体过程我们可以来分析下:

执行第三行时,会先执行multipliers函数,然后执行函数中的列表解析式。在每一次迭代的时候都会生成一个匿名函数(这里只是定义)作为元素。然后回到第三行,遍历返回的列表中的匿名函数,传入参数2并执行。

此时函数类似于这样:

def noname(x):
    return i * x

 
 
  • 1
  • 2

这里的原因就是在python中,相对而言的局部变量绑定的是值,非局部变量绑定的是空间,而不是值得本身,在上面得例子中,for循环得i,相对于lambda函数而言,是全局变量,所以绑定得是空间,最终导lambda函数运行得时候,读取得是最后保存i的值,解决办法是通过函数的默认赋值,达到保存临时变量的作用。

def multipliers():
    return [lambda x,i=i : x*i for i in range(4)]

print([m(2) for m in multipliers()])

  • 1
  • 2
  • 3
  • 4

lambda函数精简,在使用的过程一定要小心这个坑!

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值