dongzhan1492 2018-07-29 07:16
浏览 247
已采纳

函数列表中的Golang函数具有最后一个值

In the following snippet I create a func list that should, the way I see it, contain funcs that print the numbers 0 1 2.

package main

import "fmt"

func main() {
    flist := make([]func(), 0)
    for i := 0; i < 3; i++ {
        flist = append(flist, func() { fmt.Printf("%d ", i) })
    }
    for j := 0; j < 3; j++ {
        flist[j]()
    }
}

However, all funcs in flist are the same main.main.func1 and the output is

3 3 3

even though i never reaches the value 3 in the loop. This would make me thing the func is simply taking the address of i, creating the same func every time but that i is out of scope when the funcs are called later.

What am I missing here?

  • 写回答

1条回答 默认 最新

  • doupuzhimuhan9216 2018-07-29 07:25
    关注

    If we think of what code is generated for those functions, it is identical indeed: the same variable is used every time. Even though i is out of scope after the loop, it becomes the part of the closures and its latest value is seen by the function when they run. By the moment the loop exits, it's 3. If you want to bind a closure to the particular value and guard it from further changes, you may create a narrow-scoped variable, like this:

    package main
    
    import "fmt"
    
    func main() {
        flist := make([]func(), 0)
        for i := 0; i < 3; i++ {
            j := i  // a different variable for each iteration
            flist = append(flist, func() { fmt.Printf("%d ", j) })
        }
        for j := 0; j < 3; j++ {
            flist[j]()
        }
    } 
    
    本回答被题主选为最佳回答 , 对您是否有帮助呢?
    评论

报告相同问题?

悬赏问题

  • ¥15 metadata提取的PDF元数据,如何转换为一个Excel
  • ¥15 关于arduino编程toCharArray()函数的使用
  • ¥100 vc++混合CEF采用CLR方式编译报错
  • ¥15 coze 的插件输入飞书多维表格 app_token 后一直显示错误,如何解决?
  • ¥15 vite+vue3+plyr播放本地public文件夹下视频无法加载
  • ¥15 c#逐行读取txt文本,但是每一行里面数据之间空格数量不同
  • ¥50 如何openEuler 22.03上安装配置drbd
  • ¥20 ING91680C BLE5.3 芯片怎么实现串口收发数据
  • ¥15 无线连接树莓派,无法执行update,如何解决?(相关搜索:软件下载)
  • ¥15 Windows11, backspace, enter, space键失灵