go 中的锁类型包括互斥锁(mutex)、读写锁(rwmutex)、条件变量(cond)和原子操作,选择合适锁类型时应考虑并发性、读写频率和竞争可能性。
Go中的锁类型及其选择
并发编程中,锁是一个至关重要的机制,用于处理共享内存的并发访问。Go语言提供了几种锁类型,每种锁类型都有其独特的特性和适用场景。
锁类型
立即学习“go语言免费学习笔记(深入)”;
Go语言中常见的锁类型包括:
-
互斥锁(Mutex):这是一个最基本的锁,允许一次只能有一个 goroutine 访问共享资源。
var lock sync.Mutex func SomeFunc() { lock.Lock() defer lock.Unlock() // 对共享资源的访问 }
读写锁(RWMutex):这是一个更高级的锁,允许多个 goroutine 同时读取共享资源,但一次只能有一个 goroutine 写入共享资源。
var lock sync.RWMutex func SomeFunc() { lock.RLock() defer lock.RUnlock() // 读共享资源 } func OtherFunc() { lock.Lock() defer lock.Unlock() // 写共享资源 }
条件变量(Cond):这是一种特殊类型的锁,用于 goroutine 之间的通信和同步。它允许 goroutine 等待某个条件满足后才继续执行。
var cond *sync.Cond func SomeFunc() { cond.L.Lock() defer cond.L.Unlock() for { cond.Wait() // 条件满足后执行 } } func OtherFunc() { cond.L.Lock() defer cond.L.Unlock() cond.Broadcast() }
原子操作:这些操作直接操作内存位置,并确保操作原子地执行,防止出现竞争条件。
var count int32 func IncCount() { atomic.AddInt32(&count, 1) }
选择合适的锁
在选择锁类型时,考虑以下因素:
- 并发性:预计同时访问共享资源的 goroutine 数量。
- 读写频率:资源被读写的频率。
- 竞争可能性:资源发生竞争的可能性。
实战案例
考虑一个简单的银行账户:
type Account struct {
balance int
sync.Mutex
}
func (a *Account) Deposit(amount int) {
a.Lock()
a.balance += amount
a.Unlock()
}
func (a *Account) Withdraw(amount int) {
a.Lock()
a.balance -= amount
a.Unlock()
}
对于这个例子,我们选择了互斥锁,因为我们希望一次只能有一个 goroutine 更新账户余额。