Go 原子操作

作者: EasyHacking | 来源:发表于2017-11-28 21:21 被阅读0次

本文讲解 golang 中 sync.atomic 的常见操作

atomic 提供的原子操作能够确保任一时刻只有一个goroutine对变量进行操作,善用 atomic 能够避免程序中出现大量的锁操作。

atomic常见操作有:

  • 增减

  • 载入

  • 比较并交换

  • 交换

  • 存储

下面将分别介绍这些操作。

增减操作

atomic 包中提供了如下以Add为前缀的增减操作:

需要注意的是,第一个参数必须是指针类型的值,通过指针变量可以获取被操作数在内存中的地址,从而施加特殊的CPU指令,确保同一时间只有一个goroutine能够进行操作。

使用举例:

package main

import (
    "fmt"
    "sync/atomic"
    "time"
)

func main() {
    var opts int64 = 0

    for i := 0; i < 50; i++ {
        // 注意第一个参数必须是地址
        atomic.AddInt64(&opts, 3) //加操作
        //atomic.AddInt64(&opts, -1) 减操作
        time.Sleep(time.Millisecond)
    }

    time.Sleep(time.Second)

    fmt.Println("opts: ", atomic.LoadInt64(&opts))
}

载入操作

atomic 包中提供了如下以Load为前缀的增减操作:

载入操作能够保证原子的读变量的值,当读取的时候,任何其他CPU操作都无法对该变量进行读写,其实现机制收到底层硬件的支持。见上述例子中的 atomic.LoadInt64(&opts)

比较并交换

该操作简称 CAS(Compare And Swap)。 这类操作的前缀为 CompareAndSwap :

该操作在进行交换前首先确保变量的值未被更改,即仍然保持参数 old 所记录的值,满足此前提下才进行交换操作。CAS的做法类似操作数据库时常见的乐观锁机制。

需要注意的是,当有大量的goroutine 对变量进行读写操作时,可能导致CAS操作无法成功,这时可以利用for循环多次尝试。

使用示例:

var value int64

func atomicAddOp(tmp int64) {
    for {
        oldValue := value
        if atomic.CompareAndSwapInt64(&value, oldValue, oldValue+tmp) {
            return
        }
    }
}

交换

此类操作的前缀为 Swap

相对于CAS,明显此类操作更为暴力直接,并不管变量的旧值是否被改变,直接赋予新值然后返回背替换的值。

存储

此类操作的前缀为 Store

此类操作确保了写变量的原子性,避免其他操作读到了修改变量过程中的脏数据。


v 信公众号: EasyHacking

相关文章

  • Go语言——原子操作

    Go语言——原子操作 参考: 《Go并发编程实战(第2版)》 Background 原子操作即执行过程不能被中断的...

  • Go 原子操作

    本文讲解 golang 中 sync.atomic 的常见操作 atomic 提供的原子操作能够确保任一时刻只有一...

  • Go语言 原子操作

    原子操作就是不可中断的操作,外界是看不到原子操作的中间状态,要么看到原子操作已经完成,要么看到原子操作已经结束。在...

  • GO原子操作(2)

    参数设计 原子操作函数的第一个参数值对应的都应该是那个被操作的值。比如,atomic.AddInt32函数的第一个...

  • GO原子操作(1)

    读写锁是互斥锁的优化,读写锁对共享资源的写操作和读操作则区别看待,并消除了读操作之间的互斥。条件变量主要是用于协调...

  • Go - atomic包使用及atomic.Value源码分析

    1. Go中的原子操作 原子性:一个或多个操作在CPU的执行过程中不被中断的特性,称为原子性。这些操作对外表现成一...

  • 原子类与原子操作 2022-05-05

    go变量原子操作最简单的方式就是给变量配一把锁,然后在操作这个变量的方法中都加锁放锁 如【原子int】:

  • doc.go

    概述 doc.go是包atomic的提供低级原子操作的实现,对于实现同步算法很有用包含的原子操作有5种 增或减 比...

  • golang sync.once解析

    实现原理(当前代码版本go version go1.11.4 ) 1.atomic 原子操作计数器,用于记录此On...

  • go 中的 原子操作

    Go语言中提供的原子操作都是非侵入式的,在标准库代码包sync/atomic中提供了相关的原子函数。 增或减 原子...

网友评论

    本文标题:Go 原子操作

    本文链接:https://www.haomeiwen.com/subject/xzfybxtx.html