Golang:精通sync/atomic 包的Atomic 操作

在本指南中,我们将探索sync/atomic包的细节,展示如何编写更安全、更高效的并发代码。无论你是经验丰富的Gopher还是刚刚起步,你都会发现有价值的见解来提升Go编程技能。让我们一起开启原子运算的力量吧!

理解Go中的原子操作

在快节奏的并发编程世界中,原子操作是线程安全的哨兵。但是Go中的原子操作到底是什么,为什么要关心呢?让我们开始吧!
在这里插入图片描述

原子操作是不可分割的操作,对系统的其余部分来说似乎是瞬间发生的。在Go语言中,sync/atomic包提供这些操作,确保对共享变量的复杂操作不会中断。这在并发编程中是至关重要的,因为多个程序可能同时访问相同的数据。

考虑一下:你正在构建一个高流量的web应用程序,并且您需要跟踪活跃用户的数量。如果没有原子操作,您可能会遇到经典的竞争条件:

var activeUsers intfunc incrementUsers() {activeUsers++ // This is not atomic!
}

在并发环境中,这种看似无害的增量可能导致数据竞争和错误计数。下面示例采用原子操作:

import "sync/atomic"var activeUsers int64func incrementUsers() {atomic.AddInt64(&activeUsers, 1) // Atomic and safe!
}

现在,无论有多少例程调用incrementUsers(),计数总是准确的。

但是为什么要使用原子操作而不是互斥锁或其他同步方法呢?这一切都与性能有关。与基于锁的操作相比,原子操作快如闪电。事实上,Go团队的一项研究表明,对于简单的操作,原子可以比互斥体快3倍!

下面是一个简短的对比:

OperationAtomicMutex
Read2 ns6 ns
Write3 ns8 ns
CAS4 ns10 ns

(注:这些是近似值,可能会因硬件和Go版本而异)

原子操作适用于需要快速、简单同步的场景。它们适合:

  1. 计数器(像我们的活动用户示例)

  2. 标志(例如,检查进程是否完成)

  3. 简单共享状态管理

然而,需要注意的是,原子操作并不是万能的。对于复杂的数据结构或需要自动执行多个相关操作时,互斥锁或其他同步原语可能更合适。当我们深入研究同步/原子包时,请记住:能力越大责任越大。明智地使用原子操作,您的并发Go程序将以改进的性能和可靠性感谢您。

探索sync/atomic包

sync/atomic包是Go中并发编程工具的宝库。它就像一把瑞士军刀,可以安全有效地处理共享变量。让我们打开这个强大的包,看看它有什么提供!
在这里插入图片描述

sync/atomic的核心是提供低级原子内存原语,这些原语是同步算法的构建块。这些原语是用汇编语言实现的,以获得最大的效率,使它们非常快。

主要类型和功能

该包主要处理这些类型:

  • int32, int64
  • uint32, uint64
  • uintptr
  • unsafe.Pointer

对于每种类型,sync/atomic都提供了一组函数:

  1. Load:自动加载并返回变量的值。

  2. Store:自动地将值存储到变量中。

  3. Add:自动地向变量添加一个值并返回新值。

  4. Swap:自动地将一个值与一个变量交换,并返回旧值。

  5. CompareAndSwap:自动比较变量与旧值,如果它们相等,则将其与新值交换。

让我们看看这些操作:

import ("fmt""sync/atomic"
)func main() {var counter int64// Storeatomic.StoreInt64(&counter, 42)// Loadvalue := atomic.LoadInt64(&counter)fmt.Println("Counter value:", value)// AddnewValue := atomic.AddInt64(&counter, 10)fmt.Println("New counter value:", newValue)// SwapoldValue := atomic.SwapInt64(&counter, 100)fmt.Println("Old value:", oldValue, "New value:", atomic.LoadInt64(&counter))// CompareAndSwapswapped := atomic.CompareAndSwapInt64(&counter, 100, 200)fmt.Println("Swapped:", swapped, "Current value:", atomic.LoadInt64(&counter))
}

输出如下:

Counter value: 42
New counter value: 52
Old value: 52 New value: 100
Swapped: true Current value: 200

原子值操作

除了这些基本操作之外,sync/atomic还提供了Value类型,用于自动存储和加载任意值:

type Config struct {Threshold intName      string
}func main() {var configValue atomic.Value// Store a ConfigconfigValue.Store(Config{Threshold: 10, Name: "Default"})// Load the Configconfig := configValue.Load().(Config)fmt.Printf("Config: %+v\n", config)
}

这对于在并发环境中安全地更新配置值非常有用。

原子指针操作

对于更高级的用例,sync/atomic提供了指针上的原子操作:

type User struct {Name stringAge  int
}func main() {var userPtr atomic.Pointer[User]// Store a UseruserPtr.Store(&User{Name: "Alice", Age: 30})// Load the Useruser := userPtr.Load()fmt.Printf("User: %+v\n", *user)
}

这允许对复杂数据结构进行原子操作,从而启用无锁算法和数据结构。
在这里插入图片描述

性能考虑

虽然原子操作很快,但它们不是免费的。下面是一个比较原子操作和常规操作的快速基准测试:

func BenchmarkRegularIncrement(b *testing.B) {var x int64for i := 0; i < b.N; i++ {x++}
}func BenchmarkAtomicIncrement(b *testing.B) {var x int64for i := 0; i < b.N; i++ {atomic.AddInt64(&x, 1)}
}

运行这个基准测试可能会产生如下结果:

BenchmarkRegularIncrement-8    1000000000    0.3 ns/op
BenchmarkAtomicIncrement-8     100000000     12 ns/op

正如您所看到的,原子操作比常规操作慢。但是,在需要同步的并发场景中,它们通常比使用mutexes更快。

sync/atomic包是Go的并发工具包中的一个强大工具。通过理解它的功能并明智地使用它,您可以编写高效、无竞争的并发代码。

记住,能力越大责任越大。明智地使用原子操作,你的Go程序将会因为性能和可靠性的提高而感谢你

实现原子计数器

原子计数器是并发编程中的基本构建块,它允许多个例程安全地增加或减少共享值。让我们探索一下如何使用sync/atomic包来实现它们。

  • 创建和初始化原子计数器

在Go中,我们通常使用int64作为原子计数器。下面是如何创建和初始化一个:

import ("sync/atomic"
)var counter int64 // Initialized to 0 by default// Or, if you want to start with a non-zero value:
counter := atomic.Int64{}
counter.Store(100)
  • 递增和递减计数器

要自动修改计数器,可以使用AddInt64函数:

// Increment
atomic.AddInt64(&counter, 1)// Decrement
atomic.AddInt64(&counter, -1)// Add or subtract any value
atomic.AddInt64(&counter, 10)
atomic.AddInt64(&counter, -5)
  • 读计数器值

要读取计数器的当前值,使用LoadInt64:

currentValue := atomic.LoadInt64(&counter)
fmt.Printf("Current counter value: %d\n", currentValue)
  • 使用原子计数器的最佳实践
  1. 使用正确的类型:对于原子计数器始终使用int64,以确保所有体系结构上的64位对齐。

  2. 避免混合访问:不要在同一个变量上混合使用原子操作和非原子操作。

  3. 考虑溢出:记住原子计数器可以溢出,就像常规整数一样。

  4. 正确使用指针:始终向原子函数传递指针。

下面是演示这些实践的完整示例:

package mainimport ("fmt""sync""sync/atomic"
)func main() {var counter int64var wg sync.WaitGroupfor i := 0; i < 1000; i++ {wg.Add(1)go func() {atomic.AddInt64(&counter, 1)wg.Done()}()}wg.Wait()fmt.Printf("Final counter value: %d\n", atomic.LoadInt64(&counter))
}

这个程序创建了1000个例程,每个例程增加一次计数器。由于原子操作,最终值将始终为1000。

处理原子布尔值

原子布尔值对于在并发程序中实现标志非常有用。虽然Go没有专用的原子布尔类型,但我们可以使用uint32来原子地表示布尔值。

  • 设置和获取原子布尔值

下面是如何处理原子布尔值:

import ("sync/atomic"
)var flag uint32// Set the flag to true
atomic.StoreUint32(&flag, 1)// Set the flag to false
atomic.StoreUint32(&flag, 0)// Check if the flag is set
if atomic.LoadUint32(&flag) == 1 {fmt.Println("Flag is set!")
} else {fmt.Println("Flag is not set.")
}
  • 在并发程序中实现标志

原子布尔值非常适合实现控制多个线程行为的标志。下面是带有停止标志的简单worker池的例子:

package mainimport ("fmt""sync""sync/atomic""time"
)func main() {var stopFlag uint32var wg sync.WaitGroup// Start 5 workersfor i := 0; i < 5; i++ {wg.Add(1)go worker(i, &stopFlag, &wg)}// Let workers run for 2 secondstime.Sleep(2 * time.Second)// Signal workers to stopatomic.StoreUint32(&stopFlag, 1)wg.Wait()fmt.Println("All workers stopped")
}func worker(id int, stopFlag *uint32, wg *sync.WaitGroup) {defer wg.Done()for {if atomic.LoadUint32(stopFlag) == 1 {fmt.Printf("Worker %d stopping\n", id)return}// Simulate some worktime.Sleep(200 * time.Millisecond)fmt.Printf("Worker %d working\n", id)}
}
  • 原子布尔变量与常规布尔变量

对布尔变量使用原子操作比常规布尔变量有一些优势:

  1. 线程安全:原子布尔值在并发环境中使用是安全的,不需要额外的同步。

  2. 性能优势:对于简单的标志,原子布尔值通常比使用互斥锁更快。

  3. 可见性保证:原子操作确保所有例程都能立即看到更改。

然而,也有取舍:

  1. 内存使用:原子布尔值使用32位而不是常规布尔值的1位。

  2. 复杂性:语法稍微复杂一些

原子交换和比较交换(CAS)

原子交换和比较与交换(CAS)操作是高级原子原语,构成了许多无锁算法的主干。这些操作允许进行比简单的加载和存储更复杂的原子更新,从而实现各种并发数据结构和算法的高效和线程安全的实现。

  • 实现原子交换操作

Swap操作自动地将变量的值与新值交换,并返回旧值。这在需要更新值的同时还需要知道其先前状态的情况下非常有用。

下面是如何使用Swap操作:

package mainimport ("fmt""sync/atomic"
)func main() {var value int64 = 100// Atomically swap the value and get the old valueoldValue := atomic.SwapInt64(&value, 200)fmt.Printf("Old value: %d, New value: %d\n", oldValue, atomic.LoadInt64(&value))
}

输出:

Old value: 100, New value: 200

交换操作可用于各种类型:

  • SwapInt32, SwapInt64
  • SwapUint32, SwapUint64
  • SwapUintptr
  • SwapPointer

理解比较与交换(CAS)

比较与交换(CAS)是一种更强大的原语,它支持仅在值与预期值匹配时更新值。这个操作是许多无锁算法的基础。

以下是CAS的基本思想:

  1. 读取变量的当前值。

  2. 基于该值执行计算。

  3. 更新变量,但前提是它仍然具有您最初读取的值。

如果该值在步骤1和步骤3之间发生了更改,则CAS操作失败,通常需要重试该操作。

这里有一个简单的例子:

package mainimport ("fmt""sync/atomic"
)func main() {var value int64 = 100// Try to update value from 100 to 200swapped := atomic.CompareAndSwapInt64(&value, 100, 200)fmt.Printf("CAS successful: %v, New value: %d\n", swapped, atomic.LoadInt64(&value))// Try again, but it will fail because value is no longer 100swapped = atomic.CompareAndSwapInt64(&value, 100, 300)fmt.Printf("CAS successful: %v, New value: %d\n", swapped, atomic.LoadInt64(&value))
}

输出结果:

CAS successful: true, New value: 200
CAS successful: false, New value: 200

总结

我们已经游历了Go的sync/atomic包的迷人世界,揭示了原子操作的强大和巧妙。从掌握原子计数器到使用Compare-and-Swap实现无锁算法,您现在可以编写更高效、更安全的并发围棋程序了。请记住,虽然原子操作可以使代码负担过重,但它们需要仔细考虑并正确实现。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/14429.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

网络安全ITP是什么 网络安全产品ips

DS/IPS都是专门针对计算机病毒和黑客入侵而设计的网络安全设备 1、含义不同 IDS &#xff1a;入侵检测系统&#xff08;发现非法入侵只能报警不能自己过滤&#xff09; 做一个形象的比喻&#xff1a;假如防火墙是一幢大楼的门锁&#xff0c;那么IDS就是这幢大楼里的监视系统…

高速网络的未来:零拷贝Zero-Copy架构

在当今高速发展的信息技术领域&#xff0c;追求极致的性能和效率是永恒的主题。而当我们深入探索计算机系统的内部奥秘时&#xff0c;一个令人瞩目的概念 —— 零拷贝&#xff08;Zero-Copy&#xff09;架构&#xff0c;逐渐走入我们的视野。想象一下&#xff0c;在数据如洪流般…

备忘录模式

引言 当我们和朋友下棋的时候&#xff0c;我们很多情况下会发现下了一步臭棋&#xff0c;这时候就会和朋友开玩笑要悔棋&#xff0c;即撤回刚刚下的一步棋。在程序中&#xff0c;很多时候也会出错&#xff0c;我们也希望程序可以恢复出错前的状态&#xff0c;这就需要备忘录模式…

Element UI 表单源码原理

&#x1f90d; 前端开发工程师、技术日更博主、已过CET6 &#x1f368; 阿珊和她的猫_CSDN博客专家、23年度博客之星前端领域TOP1 &#x1f560; 牛客高级专题作者、打造专栏《前端面试必备》 、《2024面试高频手撕题》 &#x1f35a; 蓝桥云课签约作者、上架课程《Vue.js 和 E…

使用deepseek快速创作ppt

目录 1.在DeekSeek生成PPT脚本2.打开Kimi3.最终效果 DeepSeek作为目前最强大模型&#xff0c;其推理能力炸裂&#xff0c;但是DeepSeek官方没有提供生成PPT功能&#xff0c;如果让DeepSeek做PPT呢&#xff1f; 有个途径&#xff1a;在DeepSeek让其深度思考做出PPT脚本&#xf…

深入理解小波变换:信号处理的强大工具

引言 在科学与工程领域&#xff0c;信号处理一直是关键环节&#xff0c;傅里叶变换与小波变换作为重要的分析工具&#xff0c;在其中发挥着重要作用。本文将深入探讨小波变换&#xff0c;阐述其原理、优势以及与傅里叶变换的对比&#xff0c;并通过具体案例展示其应用价值。 一…

Kafka 入门与实战

一、Kafka 基础 1.1 创建topic kafka-topics.bat --bootstrap-server localhost:9092 --topic test --create 1.2 查看消费者偏移量位置 kafka-consumer-groups.bat --bootstrap-server localhost:9092 --describe --group test 1.3 消息的生产与发送 #生产者 kafka-cons…

WPS如何接入DeepSeek(通过第三方工具)

WPS如何接入DeepSeek 一、下载并安装OfficeAI插件二、配置OfficeAI插件三、使用DeepSeek功能 本文介绍如何通过 WPS 的第三方工具调用 DeepSeek 大模型&#xff0c;实现自动化文本扩写、校对和翻译等功能。 一、下载并安装OfficeAI插件 1、访问OfficeAI插件下载地址&#xff…

Day 32 卡玛笔记

这是基于代码随想录的每日打卡 455. 分发饼干 假设你是一位很棒的家长&#xff0c;想要给你的孩子们一些小饼干。但是&#xff0c;每个孩子最多只能给一块饼干。 对每个孩子 i&#xff0c;都有一个胃口值 g[i]&#xff0c;这是能让孩子们满足胃口的饼干的最小尺寸&#xff…

[渗透测试]热门搜索引擎推荐— — shodan篇

[渗透测试]热门搜索引擎推荐— — shodan篇 免责声明&#xff1a;本文仅用于分享渗透测试工具&#xff0c;大家使用时&#xff0c;一定需要遵守相关法律法规。 除了shodan&#xff0c;还有很多其他热门的&#xff0c;比如&#xff1a;fofa、奇安信的鹰图、钟馗之眼等&#xff0…

BUU30 [网鼎杯 2018]Fakebook1

是一个登录界面&#xff0c;我们先注册一个试试&#xff1a; 用dirsearch扫描出来robots.txt&#xff0c;也发现了flag.php&#xff0c;并下载user.php.bak 源代码内容&#xff1a; <?phpclass UserInfo {public $name "";public $age 0;public $blog &quo…

索引失效的场景

chatGpt 7. 使用 DISTINCT 或 GROUP BY 当查询中涉及 DISTINCT 或 GROUP BY 时&#xff0c;如果查询没有合适的索引支持&#xff0c;可能会导致性能问题&#xff0c;虽然不完全是索引失效&#xff0c;但会影响查询效率。 sql SELECT DISTINCT department_id FROM employees;…

3D数字化营销:重塑家居电商新生态

随着电商的蓬勃发展&#xff0c;网上订购家具已成为众多消费者的首选。然而&#xff0c;线上选购家具的诸多挑战&#xff0c;如风格不匹配、尺寸不合适、定制效果不如预期以及退换货不便等&#xff0c;一直困扰着消费者。为解决这些问题&#xff0c;家居行业急需一种全新的展示…

论文阅读--LlaVA

数据 使用GPT-4&#xff0c;根据现有的图片对数据&#xff08;image-pair data&#xff09;收集指令跟随数据。作者团队收集了158,000个独特的语言-图像指令遵循样本&#xff0c;其中包括58,000个对话样本、23,000个详细描述样本和77,000个复杂推理样本 以图像描述为例&#x…

【R语言】apply函数族

在R语言中使用循环操作时是使用自身来实现的&#xff0c;效率较低。所以R语言有一个符合其统计语言出身的特点&#xff1a;向量化。R语言中的向量化运用了底层的C语言&#xff0c;而C语言的效率比高层的R语言的效率高。 apply函数族主要是为了解决数据向量化运算的问题&#x…

归一化与伪彩:LabVIEW图像处理的区别

在LabVIEW的图像处理领域&#xff0c;归一化&#xff08;Normalization&#xff09;和伪彩&#xff08;Pseudo-coloring&#xff09;是两个不同的概念&#xff0c;虽然它们都涉及图像像素值的调整&#xff0c;但目的和实现方式截然不同。归一化用于调整像素值的范围&#xff0c…

【3分钟极速部署】在本地快速部署deepseek

第一步&#xff0c;找到网站&#xff0c;下载&#xff1a; 首先找到Ollama &#xff0c; 根据自己的电脑下载对应的版本 。 我个人用的是Windows 我就先尝试用Windows版本了 &#xff0c;文件不是很大&#xff0c;下载也比较的快 第二部就是安装了 &#xff1a; 安装完成后提示…

论文阅读:MGMAE : Motion Guided Masking for Video Masked Autoencoding

MGMAE:Motion Guided Masking for Video Masked Autoencoding Abstract 掩蔽自编码&#xff08;Masked Autoencoding&#xff09;在自监督视频表示学习中展现了出色的表现。时间冗余导致了VideoMAE中高掩蔽比率和定制的掩蔽策略。本文旨在通过引入运动引导掩蔽策略&#xff0…

【Ai】--- 可视化 DeepSeek-r1 接入 Chatbox(超详细)

在编程的艺术世界里&#xff0c;代码和灵感需要寻找到最佳的交融点&#xff0c;才能打造出令人为之惊叹的作品。而在这座秋知叶i博客的殿堂里&#xff0c;我们将共同追寻这种完美结合&#xff0c;为未来的世界留下属于我们的独特印记。 【Ai】--- 可视化 DeepSeek-r1 接入 Chat…

P1049 装箱问题(dp)

#include<bits/stdc.h> using namespace std;int main() {int v,n;cin>>v>>n;int a[30];int dp[20005];for(int i0;i<n;i){cin>>a[i];}memset(dp,0,sizeof(dp));// 设置所有元素为0&#xff0c;表示最大体积为0for(int i0;i<n;i){for(int jv;j&…