登录
首页 >  Golang >  Go问答

为什么这两个 for 循环变体会给我不同的行为?

来源:Golang技术栈

时间:2023-03-08 08:03:58 141浏览 收藏

亲爱的编程学习爱好者,如果你点开了这篇文章,说明你对《为什么这两个 for 循环变体会给我不同的行为?》很感兴趣。本篇文章就来给大家详细解析一下,主要介绍一下golang,希望所有认真读完的童鞋们,都有实质性的提高。

问题内容

我在我的程序中看到了与我程序中的这个特定循环相关的不同行为,但我不确定我是否理解它为什么会这样。

//global variable
var cmds = []string {
    "create",
    "delete",
    "update",
}

func loop1() {

    actions := make(map[string]func())

    for _, cmd := range cmds {
        actions[cmd] = func() {
            fmt.Println(cmd)
        }
    }
    for _, action := range actions {
        action()
    }
}
func loop2() {

    actions := make(map[string]func())

    for i, cmd := range cmds {
        command := cmds[i]
        actions[cmd] = func() {
            fmt.Println(command)
        }
    }
    for _, action := range actions {
        action()
    }
}

的输出loop1()

update
update
update

的输出loop2()

delete
update
create

我上网查找并阅读以下内容

在切片上进行测距时,每次迭代都会返回两个值。第一个是索引,第二个是该索引处元素的副本

它说的是一个副本,那么这是否意味着它返回了一个字符串的副本,但它实际上是一个指向变量的指针cmd?在这种情况下,任何对cmd循环结束时的引用都将实际引用数组中的最后一个元素,例如update?这是否意味着在使用该方法时数组的元素应该始终由它们的索引引用range,以及使用它返回的元素的用例是什么,因为它总是更新指针?

正确答案

问题loop1()在于您将函数文字存储在actions引用 循环变量 cmd的映射中。这个循环变量只有一个实例,所以当你在循环之后调用存储在actionsmap 中的函数时,all 将引用这个单个循环变量(因为函数/闭包仍然有对它的引用而保留它),但是它 在执行 时的值将是for循环设置的最后一个值,即cmds切片中的最后一个值(即"update",因此您将看到"update"打印了 3 次)。

一个简单的解决方法是制作此循环变量的副本,因此每次迭代,每个函数字面量都会有自己的副本,该副本与循环变量“分离”:

func loop1() {
    actions := make(map[string]func())

    for _, cmd := range cmds {
        cmd2 := cmd
        actions[cmd] = func() {
            fmt.Println(cmd2) // Refer to the detached, copy variable!
        }
    }
    for _, action := range actions {
        action()
    }
}

有了这个,输出(在Go Playgroundloop1()上试试):

update
create
delete

这不是 的问题for ... range,这是因为闭包引用同一个变量,并且您不会立即使用变量的值,只有在循环之后。当你打印这个变量的值时,所有的都打印相同的,最后一个值。

另请参阅此可能的重复项:[Golang: Register multiple routes using range for loop slices/map](https://stackoverflow.com/questions/44044245/golang-register- multiple-routes-using-range-for-loop-slices-map/44045012#44045012)

理论要掌握,实操不能落!以上关于《为什么这两个 for 循环变体会给我不同的行为?》的详细介绍,大家都掌握了吧!如果想要继续提升自己的能力,那么就来关注golang学习网公众号吧!

声明:本文转载于:Golang技术栈 如有侵犯,请联系study_golang@163.com删除
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>