首页 > 后端开发 > Golang > 正文

为什么 wg.Add() 放置对于保证 Go 中的同步至关重要?

DDD
发布: 2024-10-27 20:40:03
原创
637 人浏览过

Why is wg.Add() Placement Crucial for Guaranteed Synchronization in Go?

为保证同步而定位 wg.Add()

问题

在下面的 Go 代码中,变量 v 通常小于预期,因为wg.Add() 的位置不正确。

<code class="go">var wg sync.WaitGroup
var v int32 = 0
for i := 0; i < 100; i++ {
    go func() {
        wg.Add(1) // Wrong place
        atomic.AddInt32(&v, 1)
        wg.Done()
    }
}
wg.Wait()
fmt.Println(v)</code>
登录后复制

解决方案

wg.Add() 应始终在启动相应的 Goroutine 之前调用,以确保主 Goroutine 阻塞,直到所有goroutine 已执行 wg.Done().

<code class="go">var wg sync.WaitGroup
var v int32 = 0
for i := 0; i < 100; i++ {
    wg.Add(1)
    go func() {
        atomic.AddInt32(&v, 1)
        wg.Done()
    }
}
wg.Wait()
fmt.Println(v)</code>
登录后复制

说明

当 wg.Add() 放在 goroutine 内部时,不能保证主 goroutine 会阻塞,直到所有 goroutine 完成。这是因为 goroutine 是并发运行的,执行顺序是不确定的。

通过将 wg.Add() 放在 goroutine 之前,我们可以确保主 goroutine 在到达之前调用 wg.Add() 100 次wg.Wait().这可以确保主 goroutine 阻塞,直到所有 100 个 goroutine 都调用了 wg.Done(),从而导致 v 始终等于 100。

最佳实践

使用sync.WaitGroup 时,请遵循这些指南:

  • 在使用 go 启动新的 goroutine 之前,先在“原始”goroutine 中调用 wg.Add()。
  • 使用 defer 语句调用 wg.Done() 以确保即使 goroutine 发生恐慌,它也会被调用。
  • 将 wg 传递给其他函数时,传递一个指针以避免创建 wg 结构的副本。

以上是为什么 wg.Add() 放置对于保证 Go 中的同步至关重要?的详细内容。更多信息请关注PHP中文网其他相关文章!

来源:php.cn
本站声明
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn
热门教程
更多>
最新下载
更多>
网站特效
网站源码
网站素材
前端模板
关于我们 免责声明 Sitemap
PHP中文网:公益在线PHP培训,帮助PHP学习者快速成长!