doumo0206 2017-12-30 19:34
浏览 80
已采纳

如何使用WaitGroup确保goroutine在for循环中完成?

I'm running a function in a goroutine each time a for-loop iterates, and I'm using sync.WaitGroup to make sure the goroutines all finish. However, I'm getting weird behavior testing the concurrency with counters. In the example below, I attempt to keep track of the thread count using 4 different techniques (w, x, y, z), and get 4 different results. The only result I understand is x, since it is incremented in the for-loop itself. What am I missing here?

package main

import "fmt"
import "sync"

var w = 0

func main() {
  x := 0
  y := 0
  z := 0
  var wg sync.WaitGroup
  for i := 0; i < 10000; i++ {
    wg.Add(1)
    x++
    go func() {
      z++
      test(&y)
      wg.Done()
    }()
  }
  wg.Wait()
  fmt.Println(w, x, y, z) // 8947 10000 8831 8816
}

func test(y *int) {
  w++
  *y++
}
  • 写回答

1条回答 默认 最新

  • duandang2838 2017-12-30 20:18
    关注

    The sync.Waitgroup is working as expected. w, y and z will not reach 10000 because multiple goroutines are incrementing them concurrently, and Go's increment is not concurrent-safe: it is implemented as a normal fetch-increment-reassign operation.

    You have two options.

    option 1: mutex

    type incrementer struct {
        sync.Mutex
        i int
    }
    
    func (i *incrementer) Add(n int) {
        i.Lock()
        defer i.Unlock()
        i.i += n
    }
    

    and use this type for w, y and z.

    Full example: https://play.golang.org/p/6wWUK2xnOCW

    option 2: sync.atomic

    var w int32 = 0
    
    go func(){
        // in the loop
        atomic.AddInt32(&w, 1)
    
    }()
    

    Full example: https://play.golang.org/p/oUCGgKYC1-Y

    本回答被题主选为最佳回答 , 对您是否有帮助呢?
    评论

报告相同问题?

悬赏问题

  • ¥15 乌班图ip地址配置及远程SSH
  • ¥15 怎么让点阵屏显示静态爱心,用keiluVision5写出让点阵屏显示静态爱心的代码,越快越好
  • ¥15 PSPICE制作一个加法器
  • ¥15 javaweb项目无法正常跳转
  • ¥15 VMBox虚拟机无法访问
  • ¥15 skd显示找不到头文件
  • ¥15 机器视觉中图片中长度与真实长度的关系
  • ¥15 fastreport table 怎么只让每页的最下面和最顶部有横线
  • ¥15 java 的protected权限 ,问题在注释里
  • ¥15 这个是哪里有问题啊?