闭包中引用循环变量的问题

今天在看Go的语法,突然想对比一下python和Go的闭包机制的区别。

假如闭包调用循环变量,在Go中的实现方法为,循环中用一个新的变量保存这个循环变量的值,具体实现代码如下:

//Go
package main

import (
	"fmt"
	"reflect"
)

func main() {
	var funcArray = count()
	for index := range funcArray {
		funcArray[index]()
	}
}

func count() []func() {
	var a = make([]func(), 3)
	for index := range a {
		indexCopy := index
		a[index] = func() {
			fmt.Println(indexCopy)
		}
	}
	return a
}

三次循环中产生了三个新的indexCopy的量,于是这个程序最后打印出0, 1, 2.

但是在python中,必须这样实现:

#python

def count():
    l = []
    for i in range(3):
        def f(i):
            def g():
                return i
            return g
        l.append(f(i))
    return l

f1, f2, f3 = count()
print f1()
print f2()
print f3()

先构造一个函数f(i)把i的值固定住,加入list中,实现引用循环变量的功能。

这时候问题来了,为什么python不能像go一样,新建一个循环变量的拷贝,像下面这样呢:

#python

def count():
    l = []
    for i in range(3):
        v = i
        def f(i):
            return v
        l.append(f(i))
    return l
这是python gc决定的。在for的三次循环中,变量v的地址是不变的,只是每次循环分别指向了常量区不同的常量。所以到了最后三个闭包函数所保存的v地址是一样的,而且都指向了常量区中的2,而Go中则是三个闭包函数保存着三个indexCopy变量的地址。

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值