Go WaitGroup及Cond底层实现原理
怎么入门Golang编程?需要学习哪些知识点?这是新手们刚接触编程时常见的问题;下面golang学习网就来给大家整理分享一些知识点,希望能够给初学者一些帮助。本篇文章就来介绍《Go WaitGroup及Cond底层实现原理》,涉及到WaitGroup、底层原理、Cond,有需要的可以收藏一下
WaitGroup
概念
Go
标准库提供了WaitGroup
原语, 可以用它来等待一批 Goroutine 结束
底层数据结构
// A WaitGroup must not be copied after first use. type WaitGroup struct { noCopy noCopy state1 [3]uint32 }
其中 noCopy
是 golang 源码中检测禁止拷贝的技术。如果程序中有 WaitGroup 的赋值行为,使用 go vet
检查程序时,就会发现有报错。但需要注意的是,noCopy 不会影响程序正常的编译和运行。
state1
主要是存储着状态和信号量,状态维护了 2 个计数器,一个是请求计数器counter ,另外一个是等待计数器waiter(已调用 WaitGroup.Wait
的 goroutine 的个数)
当数组的首地址是处于一个8
字节对齐的位置上时,那么就将这个数组的前8
个字节作为64
位值使用表示状态,后4
个字节作为32
位值表示信号量(semaphore
);同理如果首地址没有处于8
字节对齐的位置上时,那么就将前4
个字节作为semaphore
,后8
个字节作为64
位数值。
使用方法
在WaitGroup里主要有3个方法:
WaitGroup.Add()
:可以添加或减少请求的goroutine数量,Add(n)
将会导致 counter += n
WaitGroup.Done()
:相当于Add(-1),Done()
将导致 counter -=1
,请求计数器counter为0 时通过信号量调用runtime_Semrelease
唤醒waiter线程
WaitGroup.Wait()
:会将 waiter++
,同时通过信号量调用 runtime_Semacquire(semap)
阻塞当前 goroutine
func main() { var wg sync.WaitGroup for i := 1; i <h2>Cond</h2> <h3>概念</h3> <p><code>Go</code>标准库提供了<code>Cond</code>原语,可以让 Goroutine 在满足特定条件时被阻塞和唤醒</p> <h3>底层数据结构</h3> <pre class="brush:go;">type Cond struct { noCopy noCopy // L is held while observing or changing the condition L Locker notify notifyList checker copyChecker } type notifyList struct { wait uint32 notify uint32 lock uintptr // key field of the mutex head unsafe.Pointer tail unsafe.Pointer }
主要有4
个字段:
nocopy
: golang 源码中检测禁止拷贝的技术。如果程序中有 WaitGroup 的赋值行为,使用 go vet
检查程序时,就会发现有报错,但需要注意的是,noCopy 不会影响程序正常的编译和运行
checker
:用于禁止运行期间发生拷贝,双重检查(Double check
)
L
:可以传入一个读写锁或互斥锁,当修改条件或者调用Wait
方法时需要加锁
notify
:通知链表,调用Wait()
方法的Goroutine
会放到这个链表中,从这里获取需被唤醒的Goroutine列表
使用方法
在Cond里主要有3个方法:
sync.NewCond(l Locker)
: 新建一个 sync.Cond 变量,注意该函数需要一个 Locker 作为必填参数,这是因为在cond.Wait()
中底层会涉及到 Locker 的锁操作Cond.Wait()
: 阻塞等待被唤醒,调用Wait函数前需要先加锁;并且由于Wait函数被唤醒时存在虚假唤醒等情况,导致唤醒后发现,条件依旧不成立,因此需要使用 for 语句来循环地进行等待,直到条件成立为止Cond.Signal()
: 只唤醒一个最先 Wait 的 goroutine,可以不用加锁Cond.Broadcast()
: 唤醒所有Wait的goroutine,可以不用加锁
package main import ( "fmt" "sync" "sync/atomic" "time" ) var status int64 func main() { c := sync.NewCond(&sync.Mutex{}) for i := 0; i < 10; i++ { go listen(c) } go broadcast(c) time.Sleep(1 * time.Second) } func broadcast(c *sync.Cond) { // 原子操作 atomic.StoreInt64(&status, 1) c.Broadcast() } func listen(c *sync.Cond) { c.L.Lock() for atomic.LoadInt64(&status) != 1 { c.Wait() // Wait 内部会先调用 c.L.Unlock(),来先释放锁,如果调用方不先加锁的话,会报错 } fmt.Println("listen") c.L.Unlock() }
文中关于golang的知识介绍,希望对你的学习有所帮助!若是受益匪浅,那就动动鼠标收藏这篇《Go WaitGroup及Cond底层实现原理》文章吧,也可关注golang学习网公众号了解相关技术文章。

- 上一篇
- Golang内存管理简单技巧详解

- 下一篇
- 使用Go重试机制代码更可靠
-
- 生动的皮带
- 好细啊,已收藏,感谢老哥的这篇技术文章,我会继续支持!
- 2023-04-26 09:01:39
-
- 稳重的小兔子
- 这篇文章内容出现的刚刚好,细节满满,受益颇多,码起来,关注楼主了!希望楼主能多写Golang相关的文章。
- 2023-03-01 16:44:56
-
- 包容的雪糕
- 很有用,一直没懂这个问题,但其实工作中常常有遇到...不过今天到这,看完之后很有帮助,总算是懂了,感谢作者分享技术贴!
- 2023-02-26 02:09:22
-
- 受伤的秋天
- 这篇技术文章出现的刚刚好,细节满满,太给力了,码住,关注师傅了!希望师傅能多写Golang相关的文章。
- 2023-02-16 00:00:02
-
- 懦弱的火车
- 这篇文章真及时,太全面了,真优秀,码起来,关注师傅了!希望师傅能多写Golang相关的文章。
- 2023-01-28 17:53:04
-
- 高兴的悟空
- 很详细,码住,感谢up主的这篇博文,我会继续支持!
- 2023-01-25 23:24:58
-
- 闪闪的狗
- 很好,一直没懂这个问题,但其实工作中常常有遇到...不过今天到这,看完之后很有帮助,总算是懂了,感谢博主分享技术文章!
- 2023-01-09 08:40:32
-
- 朴素的饼干
- 很有用,一直没懂这个问题,但其实工作中常常有遇到...不过今天到这,帮助很大,总算是懂了,感谢师傅分享博文!
- 2023-01-05 18:06:15
-
- 斯文的画板
- 赞 ??,一直没懂这个问题,但其实工作中常常有遇到...不过今天到这,帮助很大,总算是懂了,感谢博主分享博文!
- 2023-01-02 18:45:59
-
- 痴情的鞋垫
- 这篇技术文章真及时,好细啊,很有用,已加入收藏夹了,关注作者了!希望作者能多写Golang相关的文章。
- 2023-01-02 10:14:18
-
- 昏睡的云朵
- 这篇博文出现的刚刚好,很详细,真优秀,码起来,关注博主了!希望博主能多写Golang相关的文章。
- 2023-01-01 09:17:11
-
- Golang · Go教程 | 16秒前 |
- Golang打造高扩展UDP服务器技巧
- 248浏览 收藏
-
- Golang · Go教程 | 4分钟前 |
- GolangJSON数据处理实战教程
- 298浏览 收藏
-
- Golang · Go教程 | 6分钟前 |
- Go语言栈溢出防护详解
- 160浏览 收藏
-
- Golang · Go教程 | 9分钟前 |
- Gopprof工具链全面解析
- 184浏览 收藏
-
- Golang · Go教程 | 27分钟前 |
- Go泛型切片操作接口模式详解
- 327浏览 收藏
-
- Golang · Go教程 | 28分钟前 |
- Golang值类型接口实现限制解析
- 409浏览 收藏
-
- Golang · Go教程 | 33分钟前 |
- Linux下Go编译配置教程详解
- 399浏览 收藏
-
- Golang · Go教程 | 36分钟前 |
- Go泛型发展与1.18实战解析
- 158浏览 收藏
-
- Golang · Go教程 | 38分钟前 |
- Golangsync.Pool使用与优化技巧
- 308浏览 收藏
-
- Golang · Go教程 | 38分钟前 | golang 测试套件 Mock框架 TestMain函数 分层测试
- Golang测试套件组织指南:分层与Main函数解析
- 299浏览 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 542次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 511次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 498次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 484次学习
-
- 千音漫语
- 千音漫语,北京熠声科技倾力打造的智能声音创作助手,提供AI配音、音视频翻译、语音识别、声音克隆等强大功能,助力有声书制作、视频创作、教育培训等领域,官网:https://qianyin123.com
- 88次使用
-
- MiniWork
- MiniWork是一款智能高效的AI工具平台,专为提升工作与学习效率而设计。整合文本处理、图像生成、营销策划及运营管理等多元AI工具,提供精准智能解决方案,让复杂工作简单高效。
- 83次使用
-
- NoCode
- NoCode (nocode.cn)是领先的无代码开发平台,通过拖放、AI对话等简单操作,助您快速创建各类应用、网站与管理系统。无需编程知识,轻松实现个人生活、商业经营、企业管理多场景需求,大幅降低开发门槛,高效低成本。
- 96次使用
-
- 达医智影
- 达医智影,阿里巴巴达摩院医疗AI创新力作。全球率先利用平扫CT实现“一扫多筛”,仅一次CT扫描即可高效识别多种癌症、急症及慢病,为疾病早期发现提供智能、精准的AI影像早筛解决方案。
- 90次使用
-
- 智慧芽Eureka
- 智慧芽Eureka,专为技术创新打造的AI Agent平台。深度理解专利、研发、生物医药、材料、科创等复杂场景,通过专家级AI Agent精准执行任务,智能化工作流解放70%生产力,让您专注核心创新。
- 87次使用
-
- GosyncWaitGroup使用深入理解
- 2022-12-22 122浏览
-
- Golang底层原理解析String使用实例
- 2022-12-22 455浏览
-
- Go语言同步等待组sync.WaitGroup结构体对象方法详解
- 2022-12-30 201浏览
-
- Go底层channel实现原理及示例详解
- 2022-12-24 399浏览
-
- Go基础教程系列之WaitGroup用法实例详解
- 2023-01-07 452浏览