0

0

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

DDD

DDD

发布时间:2025-11-02 11:51:18

|

511人浏览过

|

来源于php中文网

原创

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

本文深入探讨go语言中`sync.waitgroup`的正确使用方法,特别是`wg.add()`调用时机的关键性。我们将通过示例代码分析,解释为何`wg.add()`必须在`go`语句之前执行,以及go内存模型如何确保这一顺序,从而有效避免竞态条件和潜在的程序崩溃,保障并发程序的稳定运行。

在Go语言的并发编程中,`sync.WaitGroup`是一个至关重要的同步原语,用于等待一组goroutine完成其任务。它通过一个内部计数器来工作:`Add(delta int)`方法增加计数器,`Done()`方法(等价于`Add(-1)`)减少计数器,而`Wait()`方法则会阻塞,直到计数器归零。

`sync.WaitGroup`的基本用法

以下是一个典型的`sync.WaitGroup`使用示例,它启动了多个goroutine并行执行任务,并在所有任务完成后才继续主goroutine的执行:

package main
<p>import (
"fmt"
"sync"
"time"
)</p><p>func dosomething(millisecs time.Duration, wg <em>sync.WaitGroup) {
duration := millisecs </em> time.Millisecond
time.Sleep(duration)
fmt.Println("Function in background, duration:", duration)
wg.Done() // 任务完成后,减少计数器
}</p><p>func main() {
var wg sync.WaitGroup
wg.Add(4) // 预先告知WaitGroup将有4个goroutine需要等待
go dosomething(200, &wg)
go dosomething(400, &wg)
go dosomething(150, &wg)
go dosomething(600, &wg)</p><pre class="brush:php;toolbar:false;">wg.Wait() // 阻塞直到所有goroutine都调用了wg.Done()
fmt.Println("Done")

}

在这个例子中,`main`函数启动了四个`dosomething` goroutine。`wg.Add(4)`在所有`go`语句之前被调用,确保了`WaitGroup`的计数器在任何goroutine开始执行并可能调用`wg.Done()`之前就已经被正确设置。`wg.Wait()`则会等待这四个goroutine全部完成,即计数器归零,才会打印"Done"并退出。

`wg.Add()`调用时机的关键性

上述示例中,`wg.Add(4)`在所有`go`语句之前执行是至关重要的。虽然将`wg.Add(1)`分散到每个`go`语句之前也是正确的:

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)
<pre class="brush:php;toolbar:false;">wg.Wait()
fmt.Println("Done")

}

但当已知goroutine数量时,一次性调用`wg.Add(N)`更为简洁高效。更重要的是,`wg.Add()`的调用必须在对应的`go`语句之前完成,以避免竞态条件和程序崩溃。

`WaitGroup`的内部计数器不能为负值。如果一个`wg.Done()`被调用时,计数器已经为零,程序将会发生`panic`。为了防止这种情况,我们必须确保每次`wg.Done()`被调用时,计数器都已经通过`wg.Add()`递增过。

Giiso写作机器人
Giiso写作机器人

Giiso写作机器人,让写作更简单

下载

Go内存模型与并发安全保障

Go语言的内存模型为我们提供了关于并发操作顺序的保证。理解这一点对于正确使用`sync.WaitGroup`至关重要:

  • 单个goroutine内的顺序: 在单个goroutine内部,所有语句的执行顺序看起来与它们在代码中出现的顺序一致。
  • `go`语句的发生顺序: Go内存模型明确指出,一个goroutine的执行不会在其对应的`go`语句完成之前开始。这意味着,在主goroutine中,`wg.Add()`语句的执行,保证在`go`语句启动的子goroutine开始执行其内部的`wg.Done()`之前。

因此,当我们在`go`语句之前调用`wg.Add()`时,Go的内存模型确保了以下顺序:

  1. 主goroutine执行`wg.Add(N)`,增加计数器。
  2. 主goroutine执行`go`语句,启动一个新的goroutine。
  3. 新的goroutine开始执行,并在完成任务后调用`wg.Done()`,减少计数器。

这种顺序保证了当子goroutine尝试调用`wg.Done()`时,`WaitGroup`的计数器已经被正确地初始化或递增,从而避免了计数器降至负数引发的`panic`。如果`wg.Add()`在`go`语句之后执行,就可能出现竞态条件:主goroutine可能在`wg.Add()`执行之前就启动了子goroutine,并且子goroutine可能在主goroutine执行`wg.Add()`之前就完成了任务并调用了`wg.Done()`,导致`panic`。

注意事项

  • `Add()`必须在`go`语句之前: 始终确保`wg.Add()`在启动任何可能调用`wg.Done()`的goroutine之前被调用。这是避免竞态条件和`panic`的核心规则。
  • `Done()`与`Add()`匹配: 每个`wg.Add(1)`或`wg.Add(N)`的增量都必须有对应的`wg.Done()`来抵消。通常,`wg.Done()`会放在`defer`语句中,以确保即使函数发生错误也能被调用,例如:
    func worker(id int, wg *sync.WaitGroup) {
        defer wg.Done() // 确保在函数退出时调用Done
        // ... 执行任务 ...
    }
  • 避免在被等待的goroutine中调用`Wait()`: 不要在被`WaitGroup`等待的goroutine内部调用同一个`WaitGroup`的`Wait()`方法,这会导致死锁。`Wait()`通常由主goroutine或协调goroutine调用。

总结

`sync.WaitGroup`是Go语言中实现并发协作的重要工具。正确理解和运用其`Add()`、`Done()`和`Wait()`方法,特别是`wg.Add()`的调用时机,是编写健壮、高效并发程序的关键。通过遵循Go内存模型的保证,确保`wg.Add()`在`go`语句之前执行,我们可以有效地避免竞态条件和运行时错误,从而构建出更加稳定可靠的并发应用。

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

WorkBuddy
WorkBuddy

腾讯云推出的AI原生桌面智能体工作台

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

更多
string转int
string转int

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

1051

2023.08.02

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

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

615

2024.08.29

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

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

335

2025.08.29

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

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

235

2025.08.29

Go中Type关键字的用法
Go中Type关键字的用法

Go中Type关键字的用法有定义新的类型别名或者创建新的结构体类型。本专题为大家提供Go相关的文章、下载、课程内容,供大家免费下载体验。

239

2023.09.06

go怎么实现链表
go怎么实现链表

go通过定义一个节点结构体、定义一个链表结构体、定义一些方法来操作链表、实现一个方法来删除链表中的一个节点和实现一个方法来打印链表中的所有节点的方法实现链表。

462

2023.09.25

go语言编程软件有哪些
go语言编程软件有哪些

go语言编程软件有Go编译器、Go开发环境、Go包管理器、Go测试框架、Go文档生成器、Go代码质量工具和Go性能分析工具等。本专题为大家提供go语言相关的文章、下载、课程内容,供大家免费下载体验。

265

2023.10.13

0基础如何学go语言
0基础如何学go语言

0基础学习Go语言需要分阶段进行,从基础知识到实践项目,逐步深入。php中文网给大家带来了go语言相关的教程以及文章,欢迎大家前来学习。

722

2023.10.26

TypeScript类型系统进阶与大型前端项目实践
TypeScript类型系统进阶与大型前端项目实践

本专题围绕 TypeScript 在大型前端项目中的应用展开,深入讲解类型系统设计与工程化开发方法。内容包括泛型与高级类型、类型推断机制、声明文件编写、模块化结构设计以及代码规范管理。通过真实项目案例分析,帮助开发者构建类型安全、结构清晰、易维护的前端工程体系,提高团队协作效率与代码质量。

49

2026.03.13

热门下载

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

精品课程

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

共32课时 | 6.3万人学习

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

共10课时 | 0.9万人学习

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

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