通八洲科技

如何使用Golang atomic操作实现原子性_保证多协程安全访问

日期:2025-12-30 00:00 / 作者:P粉602998670
Go 的 sync/atomic 包提供无锁原子操作,适用于基础类型线程安全读写,如计数器、状态标志;不支持复合逻辑或结构体批量更新,需注意内存对齐、全量 atomic 访问及 atomic.Value 的类型固定限制。

Go 语言的 sync/atomic 包提供了一组底层、无锁的原子操作函数,适用于对基础类型(如 int32int64uint32uint64uintptr*unsafe.Pointer)进行线程安全的读写,避免使用 sync.Mutex 带来的锁开销。它不能替代互斥锁处理复杂逻辑,但在计数器、状态标志、轻量级信号等场景下高效且安全。

适用场景与限制条件

atomic 操作仅保证单个操作的原子性(如一次加法、一次指针交换),不提供事务或复合操作的原子性。例如:atomic.AddInt64(&x, 1) 是原子的,但 x++(先读再写)不是;if x > 0 { x-- } 这类判断+修改组合也**不能**靠 atomic 自动保证安全,需配合 CompareAndSwap 手动实现。

常用原子操作示例

以下是最常使用的几类操作,均以 int64 为例(其他整型类似):

示例:安全的启动/停止标志

var isRunning int32 = 0 // 0=false, 1=true

func Start() bool { return atomic.CompareAndSwapInt32(&isRunning, 0, 1) }

func Stop() bool { return atomic.CompareAndSwapInt32(&isRunning, 1, 0) }

func IsRunning() bool { return atomic.LoadInt32(&isRunning) == 1 }

避免常见错误

使用 atomic 时容易忽略内存顺序和类型对齐问题:

atomic.Value:安全读写任意类型

对于非基础类型(如 mapstructslice),可使用 atomic.Value,它内部用接口+反射封装了类型安全的原子读写:

典型用途:配置热更新、只读缓存对象替换

var config atomic.Value

// 初始化 config.Store(&Config{Timeout: 30})

// 更新 config.Store(&Config{Timeout: 60})

// 读取(安全) c := config.Load().(*Config) fmt.Println(c.Timeout)