在并发操作中,原子操作是非常现实的问题,典型的就是多个 CPU 对同一个内存的值进行操作,如 i++,很可能两次 i++,这个 i 只增加了一次,sync/atomic 就是为了解决这个问题的。具体的实现在不同的操作系统的实现是不同的, Intel 的 CPU 架构机器上,主要是使用总线锁的方式实现的。 在 AMD 的 CPU 架构机器上就是使用 MESI 一致性协议的方式来保证原子操作。

Atomic 提供了对简单类型进行操作,这些类型包括 int32, int64, uint32, uint64, uintptr, unsafe.Pointer 共6个。这些函数的原子操作共有5种:增或减,比较并交换、载入、存储和交换

如,当我们不使用原子操作的时候:

package main

import (
   "strconv"
)

var num int64

func main() {
   num = 0
   ch := make(chan string)

   for i := 0; i < 10000; i++ {
      go add(ch, i)
   }

   for i := 0; i < 10000; i++ {
      <-ch
   }

   println(num)
}

func add(ch chan string, i int)  {
   num++
   ch <- ("add" + strconv.Itoa(i))
}

多次运行,最终得到的 num 值是不同的:

按照代码逻辑应该是需要为 10000 则是符合预期的,更改使用原子操作时,将 num++ 改为 atomic.AddInt64(&num, 1)

package main

import (
   "strconv"
   "sync/atomic"
)

var num int64

func main() {
   num = 0
   ch := make(chan string)

   for i := 0; i < 10000; i++ {
      go add(ch, i)
   }

   for i := 0; i < 10000; i++ {
      <-ch
   }

   println(num)
}

func add(ch chan string, i int)  {
   atomic.AddInt64(&num, 1)
   ch <- ("add" + strconv.Itoa(i))
}

这个时候是符合预期的