同步:重复利用等待组在先前的等待结果返回前
来源:stackoverflow
时间:2024-02-25 16:42:28 469浏览 收藏
大家好,我们又见面了啊~本文《同步:重复利用等待组在先前的等待结果返回前》的内容中将会涉及到等等。如果你正在学习Golang相关知识,欢迎关注我,以后会给大家带来更多Golang相关文章,希望我们能一起进步!下面就开始本文的正式内容~
问题内容
该代码在多个goroutines中并发运行,以下代码是从生产环境代码中提取的关键相关部分:
func check() {
.......check condition......
//skipEnsure and skipNative will not both false here
var wg sync.WaitGroup
if !skipEnsure {
wg.Add(1)
go func(wg *sync.WaitGroup) {
defer wg.Done()
dosomething1()
}(&wg)
}
if !skipNative {
wg.Add(1)
go func(wg *sync.WaitGroup) {
defer wg.Done()
dosomething2()
}(&wg)
}
wg.Wait() //panic here
}
func worker() {
......
go check()
}
func main() {
for i:=0;i<32;i++ {
go worker()
}
ch := make(chan any, 0)
<-ch
}
当 waitgroup 是全局变量时,我可以重现该问题,但不能在提供的代码中重现该问题。在生产环境运行一段时间后会出现panic。
正确答案
如果我理解正确的话,代码看起来像这样:
1 package main
2
3 import (
4 "sync"
5 "time"
6 )
7
8 var wg sync.waitgroup
9
10 func work() {
11 wg.add(1)
12 go func(wg *sync.waitgroup) {
13 defer wg.done()
14 time.sleep(1 * time.millisecond)
15 println("one done")
16 }(&wg)
17 wg.add(1)
18 go func(wg *sync.waitgroup) {
19 defer wg.done()
20 time.sleep(2 * time.millisecond)
21 println("two done")
22 }(&wg)
23 wg.wait() //panic here
24 }
25
26 func main() {
27 for i := 0; i < 10; i++ {
28 go work()
29 }
30 wg.wait()
31 println("done")
32 }
sync.waitgroup 不适用于此类用途。何时调用 wg.add 和 wg.wait 并不明显。如果您尝试使用数据竞争检测器运行它,您可以看到如下内容:
==================
warning: data race
write at 0x0000011707e8 by goroutine 6:
runtime.racewrite()
<autogenerated>:1 +0x24
main.work()
/1.go:23 +0x1b2
previous read at 0x0000011707e8 by goroutine 7:
runtime.raceread()
<autogenerated>:1 +0x24
main.work()
/1.go:11 +0x35
goroutine 6 (running) created at:
main.main()
/1.go:28 +0x32
goroutine 7 (running) created at:
main.main()
/1.go:28 +0x32
==================
found 1 data race(s)
您可以为每个 goroutine 使用自己的 waitgroup (如您的示例所示)。我无法想象有什么理由反对它。例如,此代码将正常工作,没有任何数据竞争:
package main
import (
"sync"
"time"
)
func work() {
var wg sync.WaitGroup
wg.Add(1)
go func(wg *sync.WaitGroup) {
defer wg.Done()
time.Sleep(1 * time.Millisecond)
println("one done")
}(&wg)
wg.Add(1)
go func(wg *sync.WaitGroup) {
defer wg.Done()
time.Sleep(2 * time.Millisecond)
println("two done")
}(&wg)
wg.Wait() //panic here
}
func main() {
var wg sync.WaitGroup
for i := 0; i < 10; i++ {
wg.Add(1)
go func() {
work()
wg.Done()
}()
}
wg.Wait()
println("done")
}以上就是《同步:重复利用等待组在先前的等待结果返回前》的详细内容,更多关于的资料请关注golang学习网公众号!
声明:本文转载于:stackoverflow 如有侵犯,请联系study_golang@163.com删除
相关阅读
更多>
-
502 收藏
-
502 收藏
-
501 收藏
-
501 收藏
-
501 收藏
最新阅读
更多>
-
139 收藏
-
204 收藏
-
325 收藏
-
478 收藏
-
486 收藏
-
439 收藏
-
357 收藏
-
352 收藏
-
101 收藏
-
440 收藏
-
212 收藏
-
143 收藏
课程推荐
更多>
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 立即学习 543次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 立即学习 516次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 立即学习 500次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 立即学习 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 立即学习 485次学习