多个goroutine通过指针访问同一内存会引发数据竞争,导致行为不可预测,必须使用互斥锁或通道等同步机制保证安全。

在Go语言中,指针和goroutine结合使用时,共享数据是不安全的,除非你主动采取同步措施。直接通过指针让多个goroutine访问同一块内存区域,会引发数据竞争(data race),导致程序行为不可预测。
指针与数据共享的本质
Go中的指针指向内存地址。当你把一个变量的指针传给多个goroutine,它们实际上操作的是同一个内存位置。
例如:
func main() {var data int = 0
for i := 0; i go func(ptr *int) {
*ptr++
}(&data)
}
time.Sleep(time.Second)
fmt.Println(data) // 结果很可能不是1000
}
这段代码中,多个goroutine同时对data进行递增,但由于没有同步机制,会出现竞态条件。
立即学习“go语言免费学习笔记(深入)”;
如何保证共享数据安全
要安全地在goroutine间共享指针指向的数据,必须使用同步手段:
- 使用sync.Mutex保护临界区:每次访问共享数据前加锁,访问完释放。 var mu sync.Mutex
- 使用sync/atomic进行原子操作:适用于简单类型如int、uint等的读写或增减。 atomic.AddInt32((*int32)(ptr), 1)
- 使用channel传递数据所有权:Go推荐的做法是“不要通过共享内存来通信,而应该通过通信来共享内存”。
mu.Lock()
*ptr++
mu.Unlock()
用channel把数据的控制权传递给某个goroutine,避免多方同时访问。
常见误区与建议
很多人误以为Go的runtime会自动处理并发安全,其实不会。即使是指针传递,Go也不会隐式加锁。
建议:
- 尽量避免裸指针在goroutine间共享。
- 优先使用channel协调数据访问。
- 若必须共享,务必配合Mutex或atomic操作。
- 使用
go run -race检测数据竞争,及时发现隐患。
基本上就这些。指针本身没有错,问题出在并发访问时缺乏保护。只要加上合适的同步机制,goroutine共享指针数据也可以是安全的。










