Go中指针本身不并发安全,需保护被指针访问的底层数据;多goroutine读写同一变量须用锁、原子操作或channel;atomic.Value适合写少读多的指针发布场景。

Go 中指针本身不是并发安全的
Go 的指针(*T)只是内存地址的值,它不自带同步语义。多个 goroutine 同时读写同一个 *int 指向的变量,或同时修改指针本身的值(比如 p = &x),都属于数据竞争——Go 的 race detector 会报 data race on variable。
真正需要保护的是「被指针访问的底层数据」,而不是指针变量本身(除非你也在并发更新指针值)。常见误判是以为“用了指针就天然支持并发”,其实只是把共享变得更隐蔽了。
- 如果多个 goroutine 只读
*T指向的数据,且该数据初始化后不再修改(即不可变),则无需同步 - 只要有一个 goroutine 写该数据,所有读写操作都必须加锁(
sync.Mutex)、用原子操作(atomic.StoreInt32等),或通过 channel 传递所有权 - 避免在闭包中捕获可变指针变量并启动 goroutine,例如
for _, p := range ptrs { go func() { *p = 1 }() }—— 这里所有 goroutine 共享最后一个p的地址
用 sync/atomic 安全地更新指针指向的基础类型
对于 *int32、*uint64、*unsafe.Pointer 这类固定大小的基础类型指针,Go 提供了原子操作支持。注意:原子操作只适用于指针所指向的值是机器字长对齐的整数或指针类型,不能用于 *string 或 *struct{}。
典型场景是状态标志或计数器:
立即学习“go语言免费学习笔记(深入)”;
AJAX即“Asynchronous Javascript And XML”(异步JavaScript和XML),是指一种创建交互式网页应用的网页开发技术。它不是新的编程语言,而是一种使用现有标准的新方法,最大的优点是在不重新加载整个页面的情况下,可以与服务器交换数据并更新部分网页内容,不需要任何浏览器插件,但需要用户允许JavaScript在浏览器上执行。《php中级教程之ajax技术》带你快速
var state int32 // 安全写 atomic.StoreInt32(&state, 1) // 安全读 v := atomic.LoadInt32(&state)
-
atomic.Value更通用:可安全存取任意类型(包括*MyStruct),但每次Store都是整体替换,不支持字段级更新 - 不要对
*int(非 int32/int64)用原子操作——它在 32 位系统上可能不是原子的 - 原子操作不提供内存屏障外的同步语义;若需配合其他变量一起生效,仍要考虑 happens-before 关系
用 atomic.Value 安全地发布指针引用
当需要在运行时动态切换一个全局配置指针(如 *Config),又不想加锁阻塞读,atomic.Value 是标准解法。它保证写入和读取都是原子的,且读到的一定是某次完整 Store 的结果。
var config atomic.Value // 存 *Config
func SetConfig(c *Config) {
config.Store(c)
}
func GetConfig() *Config {
return config.Load().(*Config)
}
-
Load()返回interface{},必须显式类型断言,断言失败会 panic,所以确保只存一种类型 - 适合「写少读多」场景;频繁写入会增加 GC 压力(旧值等待回收)
- 它不阻止你修改
*Config内部字段——若结构体本身可变,仍需额外同步机制(如字段加 mutex 或只存不可变副本)
避免在 channel 传输中意外共享指针数据
通过 channel 发送指针(如 chan *Task)很常见,但容易忽略:接收方拿到指针后,若直接修改其指向内容,就可能与发送方或其他接收方产生竞争。
- 推荐做法是:channel 只传值类型(
chan Task),或传不可变指针(chan *const Task不合法,但可用struct{ data []byte } + copy模拟) - 若必须传指针,应在文档或接口契约中明确「接收方不得修改」,或在发送前 deep copy(用
github.com/jinzhu/copier等工具) - 特别警惕
append:往通过 channel 收到的*[]int所指向切片追加元素,会修改原底层数组,导致多个 goroutine 无意共享同一块内存
指针在并发中最难缠的不是语法,而是共享意图是否清晰——一旦开始用指针,就得立刻回答:谁负责生命周期?谁允许修改?有没有隐式别名?这些问题没想清,加再多锁也没用。










