Golang协程同步方法 sync.WaitGroup实践

首先初始化WaitGroup,再通过Add增加计数,每个goroutine执行完调用Done,主线程调用Wait阻塞直至所有任务完成。

golang协程同步方法 sync.waitgroup实践

在Go语言中,sync.WaitGroup 是一种常用的协程同步机制,用于等待一组并发的goroutine执行完成。它特别适用于主线程需要等待多个子任务结束的场景,比如批量发起网络请求、并行处理数据等。使用 WaitGroup 可以避免程序提前退出,确保所有协程都执行完毕。

WaitGroup 核心方法

WaitGroup 提供了三个主要方法:

Add(delta int):增加计数器的值,通常用于添加要等待的goroutine数量。 Done():将计数器减1,一般在goroutine结束时调用,相当于 Add(-1)。 Wait():阻塞当前协程,直到计数器变为0。

基本使用示例

下面是一个简单的例子,展示如何使用 WaitGroup 等待多个goroutine完成:

package mainimport (    "fmt"    "sync"    "time")func worker(id int, wg *sync.WaitGroup) {    defer wg.Done() // 任务完成,计数器减1    fmt.Printf("Worker %d startingn", id)    time.Sleep(time.Second)    fmt.Printf("Worker %d donen", id)}func main() {    var wg sync.WaitGroup    for i := 1; i <= 3; i++ {        wg.Add(1)           // 每启动一个goroutine,计数器加1        go worker(i, &wg)    }    wg.Wait() // 等待所有goroutine完成    fmt.Println("All workers finished")}

输出结果:

立即学习“go语言免费学习笔记(深入)”;

Worker 1 starting
Worker 2 starting
Worker 3 starting
Worker 1 done
Worker 2 done
Worker 3 done
All workers finished

使用注意事项

虽然 WaitGroup 使用简单,但有几个关键点需要注意,避免常见错误:

避免Add负数或超出范围:在调用 Add 时传入负数会引发 panic,必须确保 Add 的值合理。 确保 Done 被调用:每个 Add 对应一次 Done,否则 Wait 会一直阻塞,导致死锁。 传递指针:WaitGroup 应以指针形式传递给goroutine,否则值拷贝会导致各个协程操作的是不同的实例。 Add 应在 go 之前调用:如果在goroutine内部才 Add,可能主线程的 Wait 已经开始,导致遗漏。

实际应用场景

WaitGroup 常用于并行任务处理。例如,从多个API获取数据:

func fetchAllData() {    var wg sync.WaitGroup    urls := []string{"http://a.com", "http://b.com", "http://c.com"}    for _, url := range urls {        wg.Add(1)        go func(u string) {            defer wg.Done()            resp, err := http.Get(u)            if err != nil {                fmt.Printf("Error: %sn", err)                return            }            defer resp.Body.Close()            fmt.Printf("Fetched %sn", u)        }(url)    }    wg.Wait()    fmt.Println("All requests completed")}

这段代码并发请求多个URL,并等待全部完成后再继续。

基本上就这些。WaitGroup 是Go中最基础也最实用的同步工具之一,掌握它的正确用法对编写可靠的并发程序至关重要。不复杂但容易忽略细节。

以上就是Golang协程同步方法 sync.WaitGroup实践的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1401344.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月15日 17:45:05
下一篇 2025年12月15日 17:45:17

相关推荐

  • Linux安装Golang指南 各发行版包管理方案

    在Linux上安装Golang首选包管理器方式,如Ubuntu/Debian用apt、Fedora用dnf、CentOS/RHEL用yum、Arch用pacman、OpenSUSE用zypper,命令简洁且自动配置环境;2. 若需最新版或多版本共存,则推荐手动下载官方二进制包并解压至/usr/loc…

    好文分享 2025年12月15日
    000
  • Golang处理JSON数据 结构体标签与序列化

    Golang通过encoding/json包利用结构体标签实现JSON序列化与反序列化,支持字段映射、忽略、omitempty省略零值、string字符串转换等标签用法,并可通过json.RawMessage、map[string]interface{}、自定义接口及流式处理等方式灵活应对数据结构不…

    2025年12月15日
    000
  • Golang解析XML文件 encoding/xml标准库

    解析XML需定义对应struct并用xml标签映射字段,通过xml.Unmarshal将XML数据解析到struct中。1. 定义struct时使用xml:”elementName”关联元素,嵌套结构用xml:”parent>child”表示;2…

    2025年12月15日
    000
  • 怎样为Golang搭建GPU加速环境 配置CUDA和OpenCL开发支持

    要在#%#$#%@%@%$#%$#%#%#$%@_21c++28409729565fc1a4d2dd92db269f项目中使用gpu加速,需配置cuda或opencl环境。1. 若使用nvidia显卡,安装对应驱动及cuda toolkit,并用go-cuda等库调用cuda函数,注意编译时链接.c…

    2025年12月15日 好文分享
    000
  • Golang如何检查依赖更新 go list检查

    答案是使用 go list -m -u all 检查依赖更新。该命令通过查询模块代理列出所有直接和间接依赖的最新可用版本,帮助开发者识别可更新的包,输出中带方括号的版本为可用更新,不带的表示已是最新;此命令仅检查不修改文件,实际更新需用 go get -u。定期检查可提升安全性、性能与可维护性,避免…

    2025年12月15日
    000
  • Golang hash哈希算法 MD5/SHA实现

    Go语言中MD5和SHA系列哈希算法由crypto/md5、crypto/sha1、crypto/sha256、crypto/sha512等包提供,用于生成固定长度摘要,适用于数据校验、文件指纹等场景;MD5生成128位哈希值,通常表示为32位十六进制字符串;示例代码展示了对字符串计算MD5、SHA…

    2025年12月15日
    000
  • Golang如何应用模板方法模式 通过接口实现算法骨架

    模板方法模式在 go 语言中通过接口和函数组合实现,其核心是定义算法骨架并延迟部分步骤实现。1. 可通过接口定义算法步骤,结合模板函数统一调用顺序;2. 不同结构体实现接口以定制具体步骤;3. 也可使用函数参数方式灵活传入各步骤逻辑;4. 嵌套结构体可用于复用通用步骤;5. 此模式适用于流程固定但部…

    2025年12月15日 好文分享
    000
  • Golang代理模式实现 控制对象访问中间层

    代理模式通过代理对象控制对真实对象的访问,常用于权限控制、日志记录等场景。1. 定义接口Service,包含DoSomething方法;2. RealService实现具体逻辑;3. ProxyService持有RealService引用并控制访问,如检查userRole是否为admin;4. 调用…

    2025年12月15日
    000
  • Golang map与指针配合 修改map元素值技巧

    Go中map元素不可取地址,因扩容可能导致元素移动,故禁止取址以防悬空指针。1. 可将值类型设为指针,如map[string]*User,通过指针修改值;2. 若值为struct,需读出后修改再写回map;3. 大结构体建议用指针避免复制开销;4. 并发操作需用sync.RWMutex或sync.M…

    2025年12月15日
    000
  • Golang如何搭建机密容器环境 使用Kata Containers安全沙箱

    答案是:通过结合go语言特性和kata containers的硬件级隔离能力,可构建安全的机密容器环境。具体步骤包括在支持虚拟化的宿主机上安装kata containers并配置containerd或cri-o运行时,使用golang编写应用并基于scratch镜像构建轻量级容器镜像,最后通过kub…

    2025年12月15日
    000
  • Golang文件读写操作 os和ioutil包对比

    os包更底层灵活,适合精细控制和大文件处理;2. ioutil包简洁但功能受限,适合小文件操作;3. Go 1.16起ioutil被弃用,推荐使用os和io包。 Golang中, os 和 ioutil 包都提供了文件读写的功能,但它们的设计哲学和适用场景有所不同。简单来说, os 包更底层、更灵活…

    2025年12月15日
    000
  • Golang strings库常用方法 字符串操作函数详解

    Go语言strings库提供Contains、Split、Join、Trim、Replace等常用字符串处理函数,支持子串判断、分割拼接、去空格、替换、前后缀检测、大小写转换及子串查找,适用于高效安全的字符串操作。 Go语言的 strings 库提供了丰富的字符串处理函数,广泛用于日常开发中。这些函…

    2025年12月15日
    000
  • Golang持续集成环境 GitHubActions配置

    答案:配置Golang项目GitHub Actions需定义工作流文件实现自动构建测试。首先在.github/workflows下创建YAML文件,设置触发条件为main/develop分支的push或pull_request;工作流包含检出代码、配置Go环境、缓存模块、依赖整理、编译和带竞态检测的…

    2025年12月15日
    000
  • Golang代码组织规范 包划分与命名

    包的划分应遵循模块化与清晰度原则,按领域或功能划分如user、order,结合谨慎的层级划分handler、service、store,利用internal包限制内部访问,cmd目录管理可执行文件入口,通用功能独立为小而精的工具包,命名则采用简洁小写单数形式,避免复数与模糊词汇,提升代码可读性与维护…

    2025年12月15日
    000
  • Golang原子操作详解 atomic包函数使用

    Go的atomic包提供整型、指针等类型的原子操作,支持增减(Add)、加载存储(Load/Store)、比较并交换(CAS)和交换(Swap),适用于并发下计数器、标志位等轻量场景,避免锁开销。示例包括原子计数、状态控制、单例初始化与配置更新,需注意32位系统int64非原子、禁止混合普通读写及a…

    2025年12月15日
    000
  • Golang reflect反射机制 动态类型检查

    Go语言反射通过reflect包实现,可在运行时动态获取类型与值信息。使用reflect.TypeOf()和reflect.ValueOf()分别获取变量的类型和值,通过Type.Kind()和Value.Kind()判断底层类型,支持对结构体字段及标签的遍历与操作,常用于序列化、配置解析等场景。示…

    2025年12月15日
    000
  • Golang性能分析环境 pprof工具配置

    pprof能解决Go应用的CPU高占用、内存泄漏、协程阻塞、锁竞争等问题,通过在程序中导入”net/http/pprof”并启动HTTP服务,即可采集性能数据。使用时需注意生产环境安全,避免公网暴露,合理设置block和mutex采样率,区分heap与allocs内存分析,并…

    2025年12月15日
    000
  • Golang微服务错误处理 跨服务错误传递方案

    统一错误结构体设计(含Code、Message、Status等字段)实现跨服务错误传递,通过gRPC的google.rpc.Status扩展携带自定义错误详情,并在HTTP网关层映射为标准JSON响应,结合错误码集中管理与构造函数提升可维护性,确保多协议下错误信息一致可解析。 在Golang微服务架…

    2025年12月15日
    000
  • Golang桥接模式应用 抽象与实现解耦

    桥接模式通过接口与组合将抽象与实现分离,使设备与遥控器可独立扩展。定义Device接口并实现TV等具体设备,遥控器通过持有Device接口实现解耦,基础遥控器RemoteControl提供通用控制,高级遥控器AdvancedRemoteControl通过组合扩展功能,新增设备或遥控类型无需大量继承,…

    2025年12月15日
    000
  • Golang的hash哈希算法 MD5/SHA实现

    Go语言通过crypto包提供MD5、SHA系列哈希算法,适用于数据完整性校验;使用md5.New()、sha256.New()等创建哈希对象,配合io.WriteString或io.Copy处理字符串或文件;推荐SHA256以上算法以确保安全,避免MD5和SHA1用于敏感场景。 在Go语言中,标准…

    2025年12月15日
    000

发表回复

登录后才能评论
关注微信