原子操作

原子操作就是执行过程中不能被中断的操作。

Go 的标准库 sync/atomic 提供了一些实现原子操作的方法:

  • Add
  • CompareAndSwap(简称 CAS)
  • Load
  • Swap
  • Store

这些函数针对的数据类型有:

  • int32
  • int64
  • uint32
  • uint64
  • uintptr
  • unsafe 包中的 Pointer

Add 为例,上面类型对应的原子操作函数为:

  • func AddInt32(addr *int32, delta int32) (new int32)
  • func AddInt64(addr *int64, delta int64) (new int64)
  • func AddUint32(addr *uint32, delta uint32) (new uint32)
  • func AddUint64(addr *uint64, delta uint64) (new uint64)
  • func AddUintptr(addr *uintptr, delta uintptr) (new uintptr)

unsafe.Pointer 类型,并未提供进行原子加法操作的函数。

sync/atomic 包还提供了一个名为 Value 的类型,它可以被用来存储(Store)和加载(Load)任意类型的值。

它只有两个指针方法:

  • Store
  • Load

尽量不要向原子值中存储引用类型的值

var box6 atomic.Value
v6 := []int{1, 2, 3}
box6.Store(v6)
v6[1] = 4 // 此处的操作不是并发安全的

上面的代码 v6[1] = 4 绕过了原子值而进行了非并发安全的操作。可以改为:

store := func(v []int) {
    replica := make([]int, len(v))
    copy(replica, v)
    box6.Store(replica)
}
store(v6)
v6[2] = 5

使用

互斥锁与原子操作

区别:

  • 互斥锁是用来保护临界区,原子操作用于对一个变量的更新保护
  • 互斥锁由操作系统的调度器实现,原子操作由底层硬件指令直接提供支持

对于一个变量更新的保护,原子操作通常会更有效率,并且更能利用计算机多核的优势。而互斥锁保护的共享资源每次只给一个线程使用,其它线程阻塞,用完后再把资源转让给其它线程。

使用互斥锁实现并发计数:

func MutexAdd() {
	var a int32 =  0
	var wg sync.WaitGroup
	var mu sync.Mutex
	start := time.Now()
	for i := 0; i < 10000; i++ {
		wg.Add(1)
		go func() {
			defer wg.Done()
			mu.Lock()
			a += 1
			mu.Unlock()
		}()
	}
	wg.Wait()
	timeSpends := time.Now().Sub(start).Nanoseconds()
    fmt.Printf("mutex value %d, spend time: %v\n", a, timeSpends)
}

使用原子操作替换互斥锁:

func AtomicAdd() {
	var a int32 =  0
	var wg sync.WaitGroup
	start := time.Now()
	for i := 0; i < 10000; i++ {
		wg.Add(1)
		go func() {
			defer wg.Done()
			atomic.AddInt32(&a, 1)
		}()
	}
	wg.Wait()
	timeSpends := time.Now().Sub(start).Nanoseconds()
    fmt.Printf("atomic value %d, spend time: %v\n", atomic.LoadInt32(&a), timeSpends)
}

运行后得到的结果:

mutex value 10000, spend time: 5160800 
atomic value 10000, spend time: 2577300

原子操作节省了大概一半的时间。

利用 CAS 实现自旋锁

func addValue(v int32)  {
	for {
		// 在进行读取 value 的操作的过程中,其他对此值的读写操作是可以被同时进行的,那么这个读操作很可能会读取到一个只被修改了一半的数据.
		// 因此要使用原子读取
		old := atomic.LoadInt32(&value)
		if atomic.CompareAndSwapInt32(&value, old, old + v) {
			break
		}
	}
}

在高并发的情况下,单次 CAS 的执行成功率会降低,因此需要配合循环语句 for,形成一个 for+atomic 的类似自旋乐观锁。

自旋锁的使用场景

  1. 读多写少的场景,线程能够很快地获得锁,则自旋锁非常有效,因为它避免了线程调度和上下文切换的开销。如果有大量的写操作,CAS 操作无法获取到锁, 线程会在不断的自旋中消耗大量的 CPU 时间。线程需要反复尝试获取锁,而不释放 CPU,这可能导致性能下降。
  2. 自旋锁适合用在加锁粒度很小的场景,锁的持有时间非常短。如果锁定时间较长,使用自旋锁可能导致线程长时间占用 CPU。

ABA 问题

使用 CAS,会有 ABA 问题,ABA 问题是什么?

例如,一个 goroutine a 从内存位置 V 中取出 1,这时候另一个 goroutine b 也从内存位置 V 中取出 1,并且 goroutine b 将 V 位置的值更新为 0,接着又将 V 位置的值改为 1,这时候 goroutine a 进行 CAS 操作发现位置 V 的值仍然是 1,然后 goroutine a 操作成功。虽然 goroutine a 的 CAS 操 作成功,但是这个值其实已经被修改过。

可以给变量附加时间戳、版本号等信息来解决。

最后更新于