golang atomic原子操作示例詳解
sync/atomic 介紹
當我們想要對某個變量并發(fā)安全的修改,除了使用官方提供的 mutex,還可以使用 sync/atomic 包的原子操作,它能夠保證對變量的讀取或修改期間不被其他的協(xié)程所影響。
atomic 包的原子操作是通過 CPU 指令,也就是在硬件層次去實現(xiàn)的,性能較好,不需要像 mutex 那樣記錄很多狀態(tài)。 當然,mutex 不止是對變量的并發(fā)控制,更多的是對代碼塊的并發(fā)控制,2 者側(cè)重點不一樣。
sync/atomic 操作
atomic 包有幾種原子操作,主要是 Add、CompareAndSwap、Load、Store、Swap。
Add
atomic 的 Add 是針對 int 和 uint 進行原子加值的:
func AddInt32(addr *int32, delta int32) (new int32) func AddUint32(addr *uint32, delta uint32) (new uint32) func AddInt64(addr *int64, delta int64) (new int64) func AddUint64(addr *uint64, delta uint64) (new uint64) func AddUintptr(addr *uintptr, delta uintptr) (new uintptr)
CompareAndSwap
比較并交換方法實現(xiàn)了類似樂觀鎖的功能,只有原來的值和傳入的 old 值一樣,才會去修改:
func CompareAndSwapInt32(addr *int32, old, new int32) (swapped bool) func CompareAndSwapInt64(addr *int64, old, new int64) (swapped bool) func CompareAndSwapUint32(addr *uint32, old, new uint32) (swapped bool) func CompareAndSwapUint64(addr *uint64, old, new uint64) (swapped bool) func CompareAndSwapUintptr(addr *uintptr, old, new uintptr) (swapped bool) func CompareAndSwapPointer(addr *unsafe.Pointer, old, new unsafe.Pointer) (swapped bool)
需要注意的是,CompareAndSwap 有可能產(chǎn)生 ABA 現(xiàn)象發(fā)生。也就是原來的值是 A,后面被修改 B,再后面修改為 A。在這種情況下也符合了 CompareAndSwap 規(guī)則,即使中途有被改動過。
Load
Load 方法是為了防止在讀取過程中,有其他協(xié)程發(fā)起修改動作,影響了讀取結(jié)果,常用于配置項的整個讀取。
func LoadInt32(addr *int32) (val int32) func LoadInt64(addr *int64) (val int64) func LoadUint32(addr *uint32) (val uint32) func LoadUint64(addr *uint64) (val uint64) func LoadUintptr(addr *uintptr) (val uintptr) func LoadPointer(addr *unsafe.Pointer) (val unsafe.Pointer)
Store
有原子讀取,就有原子修改值,前面提到過的 Add 只適用于 int、uint 類型的增減,并沒有其他類型的修改,而 Sotre 方法通過 unsafe.Pointer 指針原子修改,來達到了對其他類型的修改。
func StoreInt32(addr *int32, val int32) func StoreInt64(addr *int64, val int64) func StoreUint32(addr *uint32, val uint32) func StoreUint64(addr *uint64, val uint64) func StoreUintptr(addr *uintptr, val uintptr) func StorePointer(addr *unsafe.Pointer, val unsafe.Pointer)
Swap
Swap 方法實現(xiàn)了對值的原子交換,不僅 int,uint 可以交換,指針也可以。
func SwapInt32(addr *int32, new int32) (old int32) func SwapInt64(addr *int64, new int64) (old int64) func SwapUint32(addr *uint32, new uint32) (old uint32) func SwapUint64(addr *uint64, new uint64) (old uint64) func SwapUintptr(addr *uintptr, new uintptr) (old uintptr) func SwapPointer(addr *unsafe.Pointer, new unsafe.Pointer) (old unsafe.Pointer)
總結(jié)
atomic 很多時候可能都沒有使用上,畢竟 mutex 的拓展性比較好,使用起來也比較友好。但這并不妨礙我們對極致性能的追求,有時候,細節(jié)決定了性能!
以上就是golang atomic原子操作示例詳解的詳細內(nèi)容,更多關(guān)于golang atomic原子操作的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
Golang反射獲取結(jié)構(gòu)體的值和修改值的代碼示例
這篇文章主要給大家介紹了golang反射獲取結(jié)構(gòu)體的值和修改值的代碼示例及演示效果,對我們的學習或工作有一定的幫助,感興趣的同學可以參考閱讀本文2023-08-08Go語言基于HTTP的內(nèi)存緩存服務(wù)的實現(xiàn)
這篇文章主要介紹了Go語言基于HTTP的內(nèi)存緩存服務(wù),本程序采用REST接口,支持設(shè)置(Set)、獲取(Get)和刪除(Del)這3個基本操作,同時還支持對緩存服務(wù)狀態(tài)進行查詢,需要的朋友可以參考下2022-08-08Golang語言JSON解碼函數(shù)Unmarshal的使用
本文主要介紹了Golang語言JSON解碼函數(shù)Unmarshal的使用,文中通過示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下2022-01-01關(guān)于go-micro與其它gRPC框架之間的通信問題及解決方法
在之前的文章中分別介紹了使用gRPC官方插件和go-micro插件開發(fā)gRPC應(yīng)用程序的方式,都能正常走通。不過當兩者混合使用的時候,互相訪問就成了問題,下面通過本文給大家講解下go-micro與gRPC框架通信問題,一起看看吧2022-04-04