当前位置:首页 > 文章 > 正文内容

Golang package atomic库

廖万里3年前 (2022-10-28)文章106482

Go语言在设计上对同步(Synchronization,数据同步和线程同步)提供大量的支持,比如 goroutine和channel同步原语,库层面有

  • sync:提供基本的同步原语(比如Mutex、RWMutex、Locker)和 工具类(Once、WaitGroup、Cond、Pool、Map)

  • sync/atomic:提供变量的原子操作(基于硬件指令 compare-and-swap)

atomic

当我们想要对某个变量并发安全的修改,除了使用官方提供的 mutex,还可以使用 sync/atomic 包的原子操作,它能够保证对变量的读取或修改期间不被其他的协程所影响。

atomic 包的原子操作是通过 CPU 指令,也就是在硬件层次去实现的,性能较好,不需要像 mutex 那样记录很多状态。 当然,mutex 不止是对变量的并发控制,更多的是对代码块的并发控制,2 者侧重点不一样。

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

比较并交换方法实现了类似乐观锁的功能,只有原来的值和传入的 old 值一样,才会去修改,

CAS 操作, 会先比较传入的地址的值是否是 old,如果是的话就尝试赋新值,如果不是的话就直接返回 false,返回 true 时表示赋值成功。


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 有可能产生ABA现象发生。也就是原来的值是 A,后面被修改 B,再后面修改为 A。在这种情况下也符合了 CompareAndSwap 规则,即使中途有被改动过。

Load

从某个地址中取值

Load 方法是为了防止在读取过程中,有其他协程发起修改动作,影响了读取结果,常用于配置项的整个读取。

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 方法实现了对值的原子交换,不仅 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)

value类型

最后一类Value 用于任意类型的值的 Store、Load,我们开始的案例就用到了这个,这是 1.4 版本之后引入的,签名的方法都只能作用于特定的类型,引入这个方法之后就可以用于任意类型了。

sync/atomic标准库包也提供了一个Value类型,以它为基的指针类型*Value拥有四个方法(见下,其中后两个是从Go 1.17开始才支持的)。Value值用来原子读取和修改任何类型的Go值。

func (*Value) Load() (x interface{})func (*Value) Store(x interface{})func (*Value) Swap(new interface{}) (old interface{})func (*Value) CompareAndSwap(old, new interface{}) (swapped bool)

CAS

在 sync/atomic 包中的源码除了 Value 之外其他的函数都是没有直接的源码的,需要去 runtime/internal/atomic 中找寻,这里为 CAS 函数为例,其他的都是大同小异

// bool Cas(int32 *val, int32 old, int32 new)// Atomically://	if(*val == old){//		*val = new;//		return 1;//	} else//		return 0;TEXT runtime∕internal∕atomic·Cas(SB),NOSPLIT,$0-17
	MOVQ	ptr+0(FP), BX
	MOVL	old+8(FP), AX
	MOVL	new+12(FP), CX
	LOCK
	CMPXCHGL	CX, 0(BX)
	SETEQ	ret+16(FP)
	RET

在注释部分写的非常清楚,这个函数主要就是先比较一下当前传入的地址的值是否和 old 值相等,如果相等,就赋值新值返回 true,如果不相等就返回 false

我们看这个具体汇编代码就可以发现,使用了LOCK 来保证操作的原子性,CMPXCHG 指令其实就是 CPU 实现的 CAS 操作。

示例对比

//Go语言在设计上对同步(Synchronization,数据同步和线程同步)提供大量的支持,比如 goroutine和channel同步原语,库层面有//////- sync:提供基本的同步原语(比如Mutex、RWMutex、Locker)和 工具类(Once、WaitGroup、Cond、Pool、Map)//- sync/atomic:提供变量的原子操作(基于硬件指令 compare-and-swap)package mainimport (	"fmt"
	"sync"
	"sync/atomic"
	"time")var (
	x  int64
	mx sync.Mutex
	wg sync.WaitGroup
)// 普通函数, 并发不安全func Add() {
	x++
	wg.Done()
}// 互斥锁, 并发安全,性能低于原子操作func MxAdd() {
	mx.Lock()
	x++
	mx.Unlock()
	wg.Done()
}// 原子操作,并发安全,性能高于互斥锁,只针对go中的一些基本数据类型使用func AmAdd() {
	atomic.AddInt64(&x, 1)
	wg.Done()
}func main() {	// 原子操作atomic包
	// 加锁操作涉及到内核态的上下文切换, 比较耗时,代价高
	// 针对基本数据类型我们还可以使用原子操作来保证并发安全
	// 因为原子操作是go语言提供的方法,我们在用户态就可以完成,因此性能比加锁操作更好
	// go语言的原子操作由内置的库,sync/atomic完成

	start := time.Now()	for i := 0; i < 10000; i++ {
		wg.Add(1)		go Add() // 普通版Add函数不是并发安全的
		//go MxAdd() // 加锁版Add函数,是并发安全的, 但是加锁性能开销大
		//go AmAdd() // 原子操作版Add函数,是并发安全的,性能优于加锁版
	}

	end := time.Now()
	wg.Wait()
	fmt.Println(x)
	fmt.Println(end.Sub(start))

}
Golang package atomic库


//Go语言在设计上对同步(Synchronization,数据同步和线程同步)提供大量的支持,比如 goroutine和channel同步原语,库层面有//////- sync:提供基本的同步原语(比如Mutex、RWMutex、Locker)和 工具类(Once、WaitGroup、Cond、Pool、Map)//- sync/atomic:提供变量的原子操作(基于硬件指令 compare-and-swap)package mainimport (	"fmt"
	"sync"
	"sync/atomic"
	"time")var (
	x  int64
	mx sync.Mutex
	wg sync.WaitGroup
)// 普通函数, 并发不安全func Add() {
	x++
	wg.Done()
}// 互斥锁, 并发安全,性能低于原子操作func MxAdd() {
	mx.Lock()
	x++
	mx.Unlock()
	wg.Done()
}// 原子操作,并发安全,性能高于互斥锁,只针对go中的一些基本数据类型使用func AmAdd() {
	atomic.AddInt64(&x, 1)
	wg.Done()
}func main() {	// 原子操作atomic包
	// 加锁操作涉及到内核态的上下文切换, 比较耗时,代价高
	// 针对基本数据类型我们还可以使用原子操作来保证并发安全
	// 因为原子操作是go语言提供的方法,我们在用户态就可以完成,因此性能比加锁操作更好
	// go语言的原子操作由内置的库,sync/atomic完成

	start := time.Now()	for i := 0; i < 10000; i++ {
		wg.Add(1)		// go Add()  // 普通版Add函数不是并发安全的
		go MxAdd() // 加锁版Add函数,是并发安全的, 但是加锁性能开销大
		//go AmAdd() // 原子操作版Add函数,是并发安全的,性能优于加锁版
	}

	end := time.Now()
	wg.Wait()
	fmt.Println(x)
	fmt.Println(end.Sub(start))

}
Golang package atomic库


//Go语言在设计上对同步(Synchronization,数据同步和线程同步)提供大量的支持,比如 goroutine和channel同步原语,库层面有//////- sync:提供基本的同步原语(比如Mutex、RWMutex、Locker)和 工具类(Once、WaitGroup、Cond、Pool、Map)//- sync/atomic:提供变量的原子操作(基于硬件指令 compare-and-swap)package mainimport (	"fmt"
	"sync"
	"sync/atomic"
	"time")var (
	x  int64
	mx sync.Mutex
	wg sync.WaitGroup
)// 普通函数, 并发不安全func Add() {
	x++
	wg.Done()
}// 互斥锁, 并发安全,性能低于原子操作func MxAdd() {
	mx.Lock()
	x++
	mx.Unlock()
	wg.Done()
}// 原子操作,并发安全,性能高于互斥锁,只针对go中的一些基本数据类型使用func AmAdd() {
	atomic.AddInt64(&x, 1)
	wg.Done()
}func main() {	// 原子操作atomic包
	// 加锁操作涉及到内核态的上下文切换, 比较耗时,代价高
	// 针对基本数据类型我们还可以使用原子操作来保证并发安全
	// 因为原子操作是go语言提供的方法,我们在用户态就可以完成,因此性能比加锁操作更好
	// go语言的原子操作由内置的库,sync/atomic完成

	start := time.Now()	for i := 0; i < 10000; i++ {
		wg.Add(1)		// go Add()  // 普通版Add函数不是并发安全的
		// go MxAdd() // 加锁版Add函数,是并发安全的, 但是加锁性能开销大
		go AmAdd() // 原子操作版Add函数,是并发安全的,性能优于加锁版
	}

	end := time.Now()
	wg.Wait()
	fmt.Println(x)
	fmt.Println(end.Sub(start))

}
Golang package atomic库


原子操作是比其它同步技术更基础的操作。原子操作是无锁的,常常直接通过CPU指令直接实现。 事实上,其它同步技术的实现常常依赖于原子操作。例如上面的的mx.lock

Golang package atomic库


atomic包提供了底层的内存操作,对于同步算法的实现很有用, 这些函数必须谨慎的保证正确使用,除了某些特殊的底层应用,使用通道或者sync包的函数/ 类型实现同步更好

atomic 很多时候可能都没有使用上,毕竟 mutex 的拓展性比较好,使用起来也比较友好。但这并不妨碍我们对极致性能的追求,有时候,细节决定了性能!

References

  1. Golang: sync.Once

  2. Synchronization(Computer Science))

  3. SIP-20 - Improved Lazy Vals Initialization

  4. sync/atomic - Go Packages

  5. Golang package sync 剖析(一): sync.Once

  6. sync/atomic标准库包中提供的原子操作

  7. Go并发编程(五) 深入理解 sync/atomic

  8. Go 语言标准库中 atomic.Value 的前世今生

  9. 深入浅出 Go - sync/atomic 源码分析


本文链接:https://www.kkkliao.cn/?id=194 转载需授权!

分享到:

版权声明:本文由廖万里的博客发布,如需转载请注明出处。


“Golang package atomic库” 的相关文章

“双11”,如何守住钱袋子?

“双11”,如何守住钱袋子?

“双十一”快到了,“剁手”的大斧已高高举起。购物车收藏83件,有的商品开启定价预售模式小姐姐的化妆品要不少钱呀,也不知比平时便宜多少最近几天,同事们都在热议“双十一”。 今年的“双十一”跟以往相比有新变化,比如,缩短预售时间、扩容购物车、差价一键退,不少平台还从晚上8点开始销售等。自从2009年开始...

马斯克倡议设台湾特别行政区,台地区专家:不希望台海冲突,反映商人普遍心声

马斯克倡议设台湾特别行政区,台地区专家:不希望台海冲突,反映商人普遍心声

 美国电动车大厂特斯拉公司创办人马斯克针对台海紧张局势,提出中国将台湾地区设为“特别行政区”方案。彰化师范大学创意创新创业研究中心主任王信文接受香港中评社访问表示,特斯拉电动车有75%的供应链来自台湾,马斯克当然不希望台海冲突,波及商业发展,这某方面也反映商人普遍心声,商人会作此呼吁,代表已嗅到战争...

一个时代终于结束了,电商行业被改写

一个时代终于结束了,电商行业被改写

如果你仔细观察近两年电商行业的新变化,你会明显地感觉到,时代一次又一次被改写。淘宝为什么被拼多多赶超了?抖音是娱乐平台,却为什么突然改做电商了?因为一个时代结束了。众所周知,传统的电商是货架电商。货架店上的本质就是把产品上到店里,然后通过搜索引擎优化或者付费推广的模式,带来流量,然后通过促销的方法,...

苹果手机打电话时如何录音

苹果手机打电话时如何录音

在国内,通话录音这项功能我们每个人都有可能使用到!因为某种意义上,可以作为证据使用。大多数的安卓手机都会支持这项功能。而苹果手机支持通话录音吗?答案是显而易见的:不支持!为什么不支持通话录音呢?对于苹果手机而言,更关注隐私的安全。通话录音在其看来是侵犯了对方的隐私,所以苹果公司不支持通话录音功能。并...

6G专利申请量比拼:美国占比35.2%,日本占比9.9%,中国排第几?

6G专利申请量比拼:美国占比35.2%,日本占比9.9%,中国排第几?

5G对于现在的人们来说已经不是什么神秘的存在,很多国家通过近些年的研究都已经逐步掌握了5G技术。不过就5G技术发展的成熟度、设备完善程度以及信号铺设范围来说,我国都是当之无愧的世界第一。但是随着科技技术的不断发展,对于6G的技术研发也被提上日程。不少国家都开始攻克6G技术,并取得了各项专利。根据有关...

步步高创始人段永平,高手有所为有所不为,35条深度思考值得收藏

步步高创始人段永平,高手有所为有所不为,35条深度思考值得收藏

段永平,一个注定在商业史无法被忽视的存在。段永平的经历可谓传奇。他是国内第一个拍下来股神巴菲特午餐的男人,那时候他还带上了现在拼多多的创始人黄铮。而这个一手创办了小霸王、步步高等著名企业,并与Vivo、OPPO、一加和拼多多有着千丝万缕联系的企业家,这位通过投资网易、腾讯和苹果而获利颇丰的投资者,也...

发表评论

访客

看不清,换一张

◎欢迎参与讨论,请在这里发表您的看法和观点。