0

0

Go 并发编程:深入理解 sync.WaitGroup 的正确使用与并发安全

霞舞

霞舞

发布时间:2025-11-02 11:59:27

|

662人浏览过

|

来源于php中文网

原创

Go 并发编程:深入理解 sync.WaitGroup 的正确使用与并发安全

本文深入探讨 go 语言中 `sync.waitgroup` 的正确使用方法,强调 `wg.add()` 必须在 `go` 语句之前调用的重要性,以避免竞态条件和程序崩溃。通过结合 go 内存模型,详细解释了 `add()` 和 `done()` 调用的时序保证,并提供了示例代码和最佳实践,帮助开发者编写健壮的并发程序。

在 Go 语言的并发编程中,`sync.WaitGroup` 是一个非常重要的同步原语,它允许我们等待一组 goroutine 完成其执行。它通过一个内部计数器来工作:`Add()` 方法增加计数器,`Done()` 方法减少计数器,而 `Wait()` 方法会阻塞,直到计数器归零。

`sync.WaitGroup` 的基本用法

以下是一个常见的 `sync.WaitGroup` 使用示例,展示了如何等待多个后台 goroutine 完成任务:

package main

import ( "fmt" "sync" "time" )

func dosomething(millisecs time.Duration, wg sync.WaitGroup) { duration := millisecs time.Millisecond time.Sleep(duration) fmt.Println("Function in background, duration:", duration) wg.Done() // 任务完成后调用 Done() }

func main() { var wg sync.WaitGroup wg.Add(4) // 预先增加计数器,表示将启动4个 goroutine go dosomething(200, &wg) go dosomething(400, &wg) go dosomething(150, &wg) go dosomething(600, &wg)

wg.Wait() // 等待所有 goroutine 完成
fmt.Println("Done")

}

上述代码的执行结果将是:

Function in background, duration: 150ms
Function in background, duration: 200ms
Function in background, duration: 400ms
Function in background, duration: 600ms
Done

这个示例清晰地展示了 `wg.Add(4)` 在启动 goroutine 之前被调用,以及每个 goroutine 在完成任务后调用 `wg.Done()`。这种模式是 `sync.WaitGroup` 的正确且推荐用法。

核心原则:`Add()` 的时机

使用 `sync.WaitGroup` 的一个关键原则是,wg.Add() 方法必须在对应的 go 语句之前被调用。这一点至关重要,它直接关系到程序的并发安全性和稳定性。

`WaitGroup` 的内部计数器从零开始。每次调用 `Add(delta int)`,计数器会增加 `delta`;每次调用 `Done()`,计数器会减少 1(相当于 `Add(-1)`)。当计数器降至零时,`Wait()` 方法才会解除阻塞。如果计数器在任何时候尝试降到零以下,`WaitGroup` 将会触发一个 panic。因此,确保 `Add()` 调用发生在 `Done()` 之前,是避免程序崩溃的关键。

并发安全与竞态条件

假设我们错误地将 `wg.Add()` 放在了 `go` 语句之后:

func main() {
    var wg sync.WaitGroup
    // 这是一个错误的示例,可能导致竞态条件甚至 panic
    go dosomething(200, &wg)
    wg.Add(1) // 如果 goroutine 启动并调用 Done() 发生在 Add() 之前,就会出问题
    // ... 其他 goroutine
    wg.Wait()
    fmt.Println("Done")
}

在这种情况下,会发生竞态条件。Go 调度器在启动 `go dosomething(...)` 后,可能在 `wg.Add(1)` 被执行之前,就调度 `dosomething` goroutine 运行。如果 `dosomething` 足够快,它可能会在 `wg.Add(1)` 执行之前调用 `wg.Done()`。此时,`WaitGroup` 的计数器仍然是 0,`Done()` 会尝试将其减为 -1,从而导致程序 panic。即使没有 panic,如果 `Wait()` 在 `Add()` 之前执行,也可能导致 `Wait()` 过早解除阻塞,而其他 goroutine 尚未完成。

NeoAgent
NeoAgent

销售易推出的AI‑CRM智能体平台

下载

Go 内存模型解析

为了理解为什么 `Add()` 必须在 `go` 语句之前,我们需要参考 Go 内存模型。Go 内存模型定义了并发程序中事件的顺序,并提供了关于内存操作可见性的保证。

  • **单 Goroutine 内的顺序**:Go 内存模型保证,在一个单独的 goroutine 内部,程序的执行顺序与代码的书写顺序一致。
  • **Goroutine 的启动**:一个 `go` 语句在它所启动的 goroutine 开始执行之前完成。这意味着,`go` 语句之前的任何操作,都保证在新的 goroutine 开始执行其代码之前完成。

结合这两点,当我们将 `wg.Add()` 放在 `go` 语句之前时,我们得到了以下保证:

  1. `wg.Add()` 操作在主 goroutine 中执行。
  2. `go` 语句在主 goroutine 中执行,并启动一个新的 goroutine。
  3. 根据 Go 内存模型,`wg.Add()` 保证在 `go` 语句之前完成。
  4. 新的 goroutine 在 `go` 语句完成之后才开始执行。
  5. 新 goroutine 中的 `wg.Done()` 操作会在其内部逻辑执行完毕后调用。

因此,`wg.Add()` 保证在 `wg.Done()` 之前发生,从而避免了竞态条件和计数器下溢的风险。

`Add()` 的灵活调用方式

虽然一次性调用 `wg.Add(N)` 是高效且推荐的方式,尤其当已知 goroutine 数量时,但也可以选择在每次启动 goroutine 之前调用 `wg.Add(1)`。例如:

func main() {
    var wg sync.WaitGroup
    wg.Add(1)
    go dosomething(200, &wg)
    wg.Add(1)
    go dosomething(400, &wg)
    wg.Add(1)
    go dosomething(150, &wg)
    wg.Add(1)
    go dosomething(600, &wg)
wg.Wait()
fmt.Println("Done")

}

这种写法在功能上是正确的,因为它同样保证了每个 `Add(1)` 都发生在对应的 `go` 语句之前。然而,当 goroutine 数量已知时,一次性调用 `wg.Add(N)` 更加简洁和高效。如果 goroutine 的数量是动态的,那么在每次启动 goroutine 之前调用 `wg.Add(1)` 则是更合适的选择。

注意事项与最佳实践

  • **`Add()` 必须在 `go` 语句之前**:这是最核心的规则,务必遵守以确保并发安全。
  • **`WaitGroup` 传递指针**:在将 `WaitGroup` 传递给 goroutine 时,必须传递其指针 (`*sync.WaitGroup`),而不是值。否则,每个 goroutine 将操作 `WaitGroup` 的副本,导致同步失败。
  • **每个 `Done()` 对应一个 `Add()`**:确保 `Done()` 的调用次数与 `Add()` 增加的计数器值相匹配,否则 `Wait()` 可能永远阻塞或导致 panic。
  • **错误处理**:在实际应用中,goroutine 内部的错误处理也很重要。`WaitGroup` 仅负责同步 goroutine 的完成,不负责错误传递。可以结合 `context` 或通道来处理错误。

总结

`sync.WaitGroup` 是 Go 语言中实现并发同步的强大工具。正确理解和使用 `wg.Add()` 的时机是编写健壮、无竞态条件并发程序的关键。通过始终确保 `wg.Add()` 在 `go` 语句之前执行,并结合对 Go 内存模型的理解,开发者可以有效地协调 goroutine 的执行,从而构建高性能且可靠的 Go 应用程序。

相关文章

编程速学教程(入门课程)
编程速学教程(入门课程)

编程怎么学习?编程怎么入门?编程在哪学?编程怎么学才快?不用担心,这里为大家提供了编程速学教程(入门课程),有需要的小伙伴保存下载就能学习啦!

下载

本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

401

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

543

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

53

2025.08.29

C++中int的含义
C++中int的含义

本专题整合了C++中int相关内容,阅读专题下面的文章了解更多详细内容。

197

2025.08.29

npd人格什么意思 npd人格有什么特征
npd人格什么意思 npd人格有什么特征

NPD(Narcissistic Personality Disorder)即自恋型人格障碍,是一种心理健康问题,特点是极度夸大自我重要性、需要过度赞美与关注,同时极度缺乏共情能力,背后常掩藏着低自尊和不安全感,影响人际关系、工作和生活,通常在青少年时期开始显现,需由专业人士诊断。

1

2026.01.26

windows安全中心怎么关闭 windows安全中心怎么执行操作
windows安全中心怎么关闭 windows安全中心怎么执行操作

关闭Windows安全中心(Windows Defender)可通过系统设置暂时关闭,或使用组策略/注册表永久关闭。最简单的方法是:进入设置 > 隐私和安全性 > Windows安全中心 > 病毒和威胁防护 > 管理设置,将实时保护等选项关闭。

0

2026.01.26

2026年春运抢票攻略大全 春运抢票攻略教你三招手【技巧】
2026年春运抢票攻略大全 春运抢票攻略教你三招手【技巧】

铁路12306提供起售时间查询、起售提醒、购票预填、候补购票及误购限时免费退票五项服务,并强调官方渠道唯一性与信息安全。

1

2026.01.26

个人所得税税率表2026 个人所得税率最新税率表
个人所得税税率表2026 个人所得税率最新税率表

以工资薪金所得为例,应纳税额 = 应纳税所得额 × 税率 - 速算扣除数。应纳税所得额 = 月度收入 - 5000 元 - 专项扣除 - 专项附加扣除 - 依法确定的其他扣除。假设某员工月工资 10000 元,专项扣除 1000 元,专项附加扣除 2000 元,当月应纳税所得额为 10000 - 5000 - 1000 - 2000 = 2000 元,对应税率为 3%,速算扣除数为 0,则当月应纳税额为 2000×3% = 60 元。

1

2026.01.26

oppo云服务官网登录入口 oppo云服务登录手机版
oppo云服务官网登录入口 oppo云服务登录手机版

oppo云服务https://cloud.oppo.com/可以在云端安全存储您的照片、视频、联系人、便签等重要数据。当您的手机数据意外丢失或者需要更换手机时,可以随时将这些存储在云端的数据快速恢复到手机中。

1

2026.01.26

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Go 教程
Go 教程

共32课时 | 4.2万人学习

Go语言实战之 GraphQL
Go语言实战之 GraphQL

共10课时 | 0.8万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号