Golang

关注公众号 jb51net

关闭
首页 > 脚本专栏 > Golang > Go syn并发同步

Go并发同步核心库syn包的使用深度指南

作者:程序员爱钓鱼

在 Go 语言中,并发是最重要的特性之一,sync 是 Go 标准库中用于 并发控制与同步 的核心工具,下面小编就和大家详细介绍一下它的具体使用吧

在 Go 语言中,并发是最重要的特性之一。通过 goroutine 可以轻松启动成千上万个并发任务,但随之而来的问题是:如何安全地共享数据、控制执行顺序以及避免竞态条件。这正是 sync 包存在的意义。

sync 是 Go 标准库中用于 并发控制与同步 的核心工具,它提供了一系列原语(primitives),用于协调多个 goroutine 之间的执行关系。常见组件包括:MutexRWMutexWaitGroupOnceCondPool 等。

相比 channel,sync 更适合处理共享内存并发模型,尤其是在需要高性能或细粒度控制时。

互斥锁:sync.Mutex

最基础的同步工具是 Mutex(互斥锁),用于保证同一时间只有一个 goroutine 可以访问共享资源。

package main

import (
	"fmt"
	"sync"
)

var (
	count int
	mu    sync.Mutex
)

func add() {
	mu.Lock()
	count++
	mu.Unlock()
}

func main() {

	for i := 0; i < 1000; i++ {
		go add()
	}

	// 简单等待(真实项目应使用 WaitGroup)
	fmt.Scanln()

	fmt.Println(count)
}

如果没有加锁,多个 goroutine 同时修改 count,会导致数据错误(竞态条件)。加锁后可以保证数据安全。

需要注意的是:

读写锁:sync.RWMutex

当读操作远多于写操作时,可以使用读写锁优化性能。

var rw sync.RWMutex

// 读
rw.RLock()
value := count
rw.RUnlock()

// 写
rw.Lock()
count++
rw.Unlock()

特点:

适用于:

等待组:sync.WaitGroup

WaitGroup 用于等待一组 goroutine 执行完成,是并发编程中最常用的工具之一。

package main

import (
	"fmt"
	"sync"
)

func worker(id int, wg *sync.WaitGroup) {
	defer wg.Done()
	fmt.Println("worker", id)
}

func main() {

	var wg sync.WaitGroup

	for i := 0; i < 5; i++ {
		wg.Add(1)
		go worker(i, &wg)
	}

	wg.Wait()

	fmt.Println("all done")
}

执行流程:

这是 Go 中控制并发任务结束的标准方式。

只执行一次:sync.Once

Once 用于保证某段代码只执行一次,常用于初始化操作。

var once sync.Once

func initConfig() {
	fmt.Println("初始化配置")
}

func main() {

	for i := 0; i < 3; i++ {
		go once.Do(initConfig)
	}

	fmt.Scanln()
}

无论调用多少次 Do(),函数只会执行一次。

适用于:

条件变量:sync.Cond

Cond 用于在 goroutine 之间进行条件通知,类似“等待-唤醒”机制。

var mu sync.Mutex
cond := sync.NewCond(&mu)

ready := false

go func() {
	cond.L.Lock()
	for !ready {
		cond.Wait()
	}
	fmt.Println("开始执行")
	cond.L.Unlock()
}()

// 模拟准备完成
cond.L.Lock()
ready = true
cond.Signal()
cond.L.Unlock()

常见方法:

Wait()    等待条件
Signal()  唤醒一个
Broadcast() 唤醒所有

适用于:

对象池:sync.Pool

sync.Pool 用于缓存临时对象,减少内存分配,提高性能。

var pool = sync.Pool{
	New: func() interface{} {
		return make([]byte, 1024)
	},
}

func main() {

	buf := pool.Get().([]byte)

	// 使用 buf

	pool.Put(buf)
}

特点:

适用于:

Map 并发安全:sync.Map

Go 原生 map 不是并发安全的,sync.Map 提供了并发安全的 map。

var m sync.Map

m.Store("key", "value")

v, ok := m.Load("key")

fmt.Println(v, ok)

常用方法:

Store
Load
Delete
Range

适用于:

sync 与 channel 对比

Go 并发有两种主流方式:

channel sync

对比:

方式特点
channel通信优先
sync共享内存控制

简单理解:

实际开发中,两者通常结合使用。

常见使用场景

在实际项目中,sync 广泛应用于:

例如实现一个并发安全计数器:

type Counter struct {
	mu sync.Mutex
	n  int
}

func (c *Counter) Add() {
	c.mu.Lock()
	defer c.mu.Unlock()
	c.n++
}

常见错误

忘记 Unlock:

mu.Lock()
// 没有 Unlock

导致死锁。

WaitGroup 使用错误:

wg.Add(1)
wg.Wait()
go func() {
	wg.Done()
}()

可能 panic。

复制 sync 对象:

mu2 := mu // 错误

sync 类型不能复制。

使用建议

实际开发中推荐:

同时注意:

总结

sync 是 Go 并发编程的核心工具库,它提供了一整套用于 协调 goroutine、保护共享数据、控制执行顺序 的机制。

核心组件包括:

在高并发系统、Web 服务、任务调度器、缓存系统等场景中,sync 都是不可或缺的基础工具。熟练掌握 sync,可以让你的 Go 程序在并发安全和性能方面达到更高水平。

以上就是Go并发同步核心库syn包的使用深度指南的详细内容,更多关于Go syn并发同步的资料请关注脚本之家其它相关文章!

您可能感兴趣的文章:
阅读全文