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

2. atomic包
方法 解释
func LoadInt32(addr int32) (val int32)
func LoadInt64(addr `
int64) (val int64)<br>func LoadUint32(addruint32) (val uint32)<br>func LoadUint64(addruint64) (val uint64)<br>func LoadUintptr(addruintptr) (val uintptr)<br>func LoadPointer(addrunsafe.Pointer`) (val 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)
写入操作
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)
修改操作
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)
交换操作
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)

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

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包的函数/类型实现同步更好

最新文章

  1. NYOJ-858下三角矩阵
  2. python递归实现折半查找
  3. Xcode添加代码块
  4. LightOJ 1236 Pairs Forming LCM (LCM 唯一分解定理 + 素数筛选)
  5. 中文乱码 jsp正常后台接收异常
  6. jqery选择器
  7. JQuery 1.3.2联动获取部门
  8. extern、static、auto、register 定义变量的不同用法
  9. Android开发(22)--seekBar采用handler消息处理操作
  10. Linux命令之初出茅庐
  11. 彻底清除Linux centos minerd木马
  12. selenium 使用
  13. C# String 与 StringBuilder
  14. 访谈:BugPhobia’s Brief Communication
  15. 洛谷P4240 毒瘤之神的考验 【莫比乌斯反演 + 分块打表】
  16. DataSet 动态添加列
  17. Alpha 冲刺报告(3/10)
  18. 2.java设计模式-抽象工厂模式
  19. Tool:Visual Studio
  20. 阿里云Docker镜像仓库(Docker Registry)

热门文章

  1. vscode 快速入门
  2. nim_duilib(5)之option
  3. Win8/Win10 Ctrl+Alt+方向键 屏幕显示翻转解决办法
  4. 【九度OJ】题目1441:人见人爱 A ^ B 解题报告
  5. 【九度OJ】题目1474:矩阵幂 解题报告
  6. 【LeetCode】496. Next Greater Element I 解题报告(Python & C++)
  7. 第七个知识点:随机性如何辅助计算和什么是BPP类问题
  8. pandas tutorial
  9. [opencv]opencv主要组件介绍
  10. [opencv]approxDP多边形逼近获取四边形轮廓信息