贝利信息

如何使用Golang实现并发安全计数器_Golang sync/atomic原子操作方法

日期:2026-01-24 00:00 / 作者:P粉602998670
不能直接用 int 类型做并发计数,因为 i++ 非原子,包含读取、加1、写回三步,多 goroutine 竞争会导致丢失更新;sync/atomic 提供 Add、Load、Store、CAS 等位宽明确的原子操作,Go 1.19+ 推荐使用 atomic.Int64 封装安全计数器。

为什么不能直接用 int 类型做并发计数

多个 goroutine 同时对一个普通 int 变量执行 ++--,结果大概率出错。因为 i++ 实际包含三步:读取、加 1、写回,中间可能被其他 goroutine 插入修改,导致丢失更新。这不是“偶尔出错”,而是只要并发足够高,就一定会复现。

sync/atomic 提供哪些原子操作函数

sync/atomicint32int64uint32uint64uintptr 和指针类型提供原子读写与运算。最常用的是:

注意:没有 atomic.AddInt,必须显式指定位宽(int32int64),否则编译报错。

完整可运行的并发安全计数器示例

下面是一个基于 atomic.Int64(Go 1.19+ 推荐)的计数器封装,比裸用 atomic.AddInt64 更安全、更易复用:

package main

import (
    "fmt"
    "sync"
    "sync/atomic"
    "time"
)

type Counter struct {
    v atomic.Int64
}

func (c *Counter) Inc() int64 {
    return c.v.Add(1)
}

func (c *Counter) Load() int64 {
    return c.v.Load()
}

func main() {
    var wg sync.WaitGroup
    var counter Counter

    for i := 0; i < 100; i++ {
        wg.Add(1)
        go func() {
            defer wg.Done()
            for j := 0; j < 1000; j++ {
                counter.Inc()
            }
        }()
    }

    wg.Wait()
    fmt.Println("Final count:", counter.Load()) // 总是输出 100000
}

如果你还在用 Go atomic.AddInt64(&v, 1) 替代 c.v.Add(1),原理相同,只是 API 更底层。

容易踩的坑和性能提示

原子操作不是万能锁替代品,用错反而引入隐蔽 bug:

真正要注意的,是位宽匹配和变量生命周期——这两点出错,程序不会 panic,但结果不可预测。