waitgroup死锁通常由add()和done()调用不匹配或重复使用导致。要避免死锁,需确保三点:1. 启动goroutine前调用add(1),并在对应goroutine末尾使用defer wg.done()保证成对执行;2. 避免在wait()之后再次调用add(),应一次性使用waitgroup,重复需求需新建实例;3. add()参数不得使计数器为负,否则引发panic。此外,golang同步机制还包括mutex、rwmutex、channel、cond和atomic,适用于不同并发场景。

WaitGroup是Golang中用于等待一组goroutine完成的同步机制。你可以把它想象成一个计数器,主goroutine等待这个计数器归零。

WaitGroup的核心在于三个方法:
Add(delta int)
Done()
Wait()
Add()
Done()
Wait()

package main
import (
"fmt"
"sync"
"time"
)
func worker(id int, wg *sync.WaitGroup) {
defer wg.Done() // 确保goroutine结束时计数器减一
fmt.Printf("Worker %d starting\n", id)
time.Sleep(time.Second) // 模拟耗时操作
fmt.Printf("Worker %d done\n", id)
}
func main() {
var wg sync.WaitGroup
for i := 1; i <= 3; i++ {
wg.Add(1) // 启动一个goroutine,计数器加一
go worker(i, &wg)
}
wg.Wait() // 等待所有goroutine完成
fmt.Println("All workers done!")
}死锁往往发生在
Add()
Done()
Add()
Done()
Add()
Wait()
立即学习“go语言免费学习笔记(深入)”;
避免死锁的关键在于确保
Add()
Done()
Add()
Done()
defer
Done()

另一种可能导致死锁的情况是在
Wait()
Add()
Add()
package main
import (
"fmt"
"sync"
)
func main() {
var wg sync.WaitGroup
wg.Add(1)
wg.Done() // 计数器变为0
wg.Add(-2) // 计数器变为-2,会panic
fmt.Println("This line will not be printed")
}因此,在使用
Add()
除了WaitGroup,Golang还提供了其他的同步机制,例如:
选择哪种同步机制取决于具体的应用场景。WaitGroup适用于等待一组goroutine完成,Mutex和RWMutex适用于保护共享资源,Channel适用于goroutine之间的通信,Cond适用于复杂的条件同步,Atomic适用于简单的原子操作。选择合适的同步机制可以提高程序的性能和可靠性。
以上就是Golang中的WaitGroup是什么 如何使用WaitGroup同步Golang并发任务的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号