本文讲解 golang 中 sync.atomic 的常见操作
atomic 提供的原子操作能够确保任一时刻只有一个goroutine对变量进行操作,善用 atomic 能够避免程序中出现大量的锁操作。
atomic常见操作有:
下面将分别介绍这些操作。
atomic 包中提供了如下以Add为前缀的增减操作:
需要注意的是,第一个参数必须是指针类型的值,通过指针变量可以获取被操作数在内存中的地址,确保同一时间只有一个goroutine能够进行操作。
先来看一个例子: 分别用“锁”和原子操作来实现多个 goroutine 对同一个变量进行累加操作。
package main import ( "fmt" "sync" "time" ) func main() { var ( mux sync.Mutex total = 0 ) for i := 0; i < 10; i++ { go func() { for { mux.Lock() total += 1 mux.Unlock() time.Sleep(time.Millisecond) } }() } time.Sleep(time.Second) fmt.Println("The total number is", atomic.LoadUint64(&total)) }
运行代码可以得到类似输出:
The total number is 7770
package main import ( "fmt" "sync/atomic" "time" ) func main() { var total int64 for i := 0; i < 10; i++ { go func() { for { atomic.AddInt64(&total, 1) time.Sleep(time.Millisecond) } }() } time.Sleep(time.Second) fmt.Println("The total number is", total) }
The total number is 7701
atomic 包中提供了如下以Load为前缀的载入操作:
载入操作能够保证原子的读变量的值,当读取的时候,任何其他 goroutine 都无法对该变量进行读写.
该操作简称 CAS(Compare And Swap)。 这类操作的前缀为 CompareAndSwap :
CompareAndSwap
该操作在进行交换前首先确保变量的值未被更改,即仍然保持参数 old 所记录的值,满足此前提下才进行交换操作。CAS的做法类似操作数据库时常见的乐观锁机制。 需要注意的是,当有大量的goroutine 对变量进行读写操作时,可能导致CAS操作无法成功,这时可以利用for循环多次尝试。
old
var value int64 func atomicAddOp(tmp int64) { for { oldValue := value if atomic.CompareAndSwapInt64(&value, oldValue, oldValue+tmp) { return } } }
此类操作的前缀为 Swap:
Swap
相对于CAS,明显此类操作更为暴力直接,并不管变量的旧值是否被改变,直接赋予新值然后返回背替换的值。
此类操作的前缀为 Store:
Store
在原子地存储某个值的过程中,任何 goroutine 都不会进行针对同一个值的读或写操作。
atomic
Copyright© 2013-2020
All Rights Reserved 京ICP备2023019179号-8