golang中的原子操作atomic包

  1. 概念
    原子操作 atomic 包
    加锁操作涉及到内核态的上下文切换,比较耗时,代价高,
    针对基本数据类型我们还可以使用原子操作来保证并发的安全,
    因为原子操作是go语言提供的方法,我们在用户态就可以完成,因此性能比加锁操作更好
    go语言的原子操作由内置的库 sync/atomic 完成

  2. 案例比较互斥锁和原子操作的性能

package main

import (
	"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))

}

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

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值