The Go Programming Language
http://golang.org/
Go Playground
Go Projects
Revel Web Framework
dyllen

1.23 版本, for-range 指针数组,闭包捕获的 range 变量为什么还是最后一个呢?

  •  
  •   dyllen · Nov 19, 2024 · 1915 views
  •   You need to sign in to view this topic
    This topic created in 563 days ago, the information mentioned may be changed or developed.
    for _, cg := range consumers {
    		log.Logger.Infof("%p", cg)
    		utils.SafeGo(func() {
    			log.Logger.Infof("%#v", cg)
    			log.Logger.Infof("%p", cg)
    			StartConsumer(ctx, cfg.Endpoint, cfg.AccessKey, cfg.SecretKey, cg)
    		})
    	}
    

    输出:

    0xc000683540
    0xc000683560
    &queue.Test2{groupName:"test2"}
    &queue.Test2{groupName:"test2"}
    0xc000683560
    0xc000683560
    

    consumers 包含两个元素,值都是指针。 1.22 版本开始 range 每次不是会创建新变量了吗?为什么我 SafeGo 里面 cg 拿到的还是最后一个的值?

    Supplement 1  ·  Nov 20, 2024
    go 1.21 版之后,go.mod 里面的 go 版本号会影响编译器,1.21 之前的就只是用来看的。
    fireice
        1
    fireice  
       Nov 19, 2024
    用匿名函数的参数传进去就 ok 了
    dyllen
        2
    dyllen  
    OP
       Nov 19, 2024
    @fireice 按道理 1.22 开始版本不需要这样做了呀,我的疑惑就是这里。
    zealot0630
        3
    zealot0630  
       Nov 19, 2024 via Android
    cg 变量从头到尾都只有一个啊,所有匿名函数捕获的变量都指向这同一个 cg
    pike0002
        4
    pike0002  
       Nov 19, 2024   ❤️ 1
    检查下 go.mod 里面的 go 的版本,如果是 1.22 之前的话新的特性就不会生效
    dyllen
        5
    dyllen  
    OP
       Nov 19, 2024
    @zealot0630 每次循环 cg 变量都是不同的,你看输出的指针地址。
    https://go.dev/play/p/kV2gxXfzVfc 你看看整个测试代码,我帖子里面的代码和这个测试的一样的,就除了 slice 的数据类型不同。
    dyllen
        6
    dyllen  
    OP
       Nov 19, 2024
    @pike0002 还真是,我一直以为 go.mod 里面那个版本只是用来看我这个项目需要的最低 go 版本,原来还会影响实际运行版本的行为。
    About   ·   Help   ·   Advertise   ·   Blog   ·   API   ·   FAQ   ·   Solana   ·   3016 Online   Highest 6679   ·     Select Language
    创意工作者们的社区
    World is powered by solitude
    VERSION: 3.9.8.5 · 33ms · UTC 06:09 · PVG 14:09 · LAX 23:09 · JFK 02:09
    ♥ Do have faith in what you're doing.