Mutex是互斥锁,用于控制多个goroutine对共享资源的访问。Go中通过sync.Mutex提供Lock和Unlock方法实现加锁与释放,确保临界区同一时间仅被一个goroutine访问。典型用法为mutex.Lock()后紧跟defer mutex.Unlock()以安全操作共享数据。例如并发计数器中,加锁可避免数据竞争,使最终结果正确。使用时需注意:避免复制含Mutex的结构体、不在锁内执行耗时操作、防止重复加锁导致死锁。读多写少场景可用sync.RWMutex提升性能,允许多个读但写独占。结合go run -race命令可检测竞态条件,确保并发安全。

在并发编程中,多个goroutine同时访问共享资源容易引发数据竞争问题。Golang通过sync包提供的Mutex(互斥锁)来保护共享数据,确保同一时间只有一个goroutine能访问临界区,从而保障数据安全。
什么是Mutex
Mutex是“Mutual Exclusion”的缩写,即互斥锁。它是一种同步原语,用于控制多个协程对共享资源的访问。在Go中,*sync.Mutex* 提供了两个主要方法:
- Lock():获取锁,如果已被其他goroutine持有,则阻塞等待
- Unlock():释放锁,必须由加锁的goroutine调用,否则可能导致程序崩溃
正确使用模式通常是:
mutex.Lock()defer mutex.Unlock()
// 操作共享数据
实际场景:并发计数器
假设我们要实现一个并发安全的计数器,多个goroutine同时增加计数。不加锁会导致结果错误。
立即学习“go语言免费学习笔记(深入)”;
非线程安全示例(可能出错):
for i := 0; i go func() {
count++ // 数据竞争
}()
}
使用Mutex修复:
var (count int
mu sync.Mutex
)
for i := 0; i go func() {
mu.Lock()
defer mu.Unlock()
count++
}()
}
运行结束后,count值为1000,结果正确。
常见使用技巧与注意事项
- 总是配对使用Lock和Unlock:推荐配合defer使用,避免忘记释放导致死锁
- 不要复制包含Mutex的结构体:复制后可能导致多个实例操作同一个锁,应使用指针传递
- 避免在持有锁时做耗时操作:如网络请求、文件读写等,会降低并发性能
- 可重入问题:Go的Mutex不支持递归加锁,同一线程重复Lock会死锁
结构体中嵌入Mutex的标准做法:
type SafeCounter {mu sync.Mutex
val int
}
func (c *SafeCounter) Inc() {
c.mu.Lock()
defer c.mu.Unlock()
c.val++
}
RWMutex:读写更高效的场景
当读多写少时,可使用*sync.RWMutex*提升性能:
- 多个读操作可以同时进行(RLock/ RUnlock)
- 写操作独占访问(Lock/ Unlock)
data map[string]string
mu sync.RWMutex
}
// 读操作
mu.RLock()
value := data[key]
mu.RUnlock()
// 写操作
mu.Lock()
data[key] = value
mu.Unlock()
基本上就这些。合理使用Mutex能有效防止数据竞争,但也要注意不要过度加锁影响性能。结合go run -race命令检测竞态条件,是开发阶段的重要保障手段。










