Golang原子操作详解 atomic包函数使用

Go的atomic包提供整型、指针等类型的原子操作,支持增减(Add)、加载存储(Load/Store)、比较并交换(CAS)和交换(Swap),适用于并发下计数器、标志位等轻量场景,避免锁开销。示例包括原子计数、状态控制、单例初始化与配置更新,需注意32位系统int64非原子、禁止混合普通读写及atomic.Value不可复制等问题。

golang原子操作详解 atomic包函数使用

在Go语言中,atomic包提供了对基本数据类型的原子操作支持,用于在并发环境下安全地读取、写入、修改共享变量,而无需使用互斥锁(

sync.Mutex

)。原子操作效率更高,适用于简单的共享状态管理,比如计数器、标志位等场景。

atomic包常用函数分类

atomic包主要支持对整型(

int32

int64

)、指针、

uint32

uint64

uintptr

bool

类型的原子操作。以下是核心函数分类及使用方式:

1. 原子增减(Add)

用于对整型变量进行原子加减操作:

立即学习“go语言免费学习笔记(深入)”;

atomic.AddInt32(&val, delta)

:对

int32

变量加

delta
atomic.AddInt64(&val, delta)

:对

int64

变量加

delta
atomic.AddUint32(&val, delta)

:对

uint32

atomic.AddUint64(&val, delta)

:对

uint64

atomic.AddUintptr

:用于指针偏移,较少使用

示例:实现一个并发安全的计数器

var counter int64

func increment() {atomic.AddInt64(&counter, 1)}

func main() {var wg sync.WaitGroupfor i := 0; i < 1000; i++ {wg.Add(1)go func() {defer wg.Done()increment()}()}wg.Wait()fmt.Println("Counter:", atomic.LoadInt64(&counter)) // 1000}

2. 原子加载与存储(Load / Store)

用于安全地读取和写入变量值,避免并发读写导致的数据竞争。

atomic.LoadInt32(&val)

:原子读取

int32
atomic.LoadInt64(&val)

:原子读取

int64
atomic.LoadUint32(&val)

:原子读取

uint32
atomic.LoadPointer(&ptr)

:原子读取指针

atomic.StoreInt32(&val, new)

:原子写入

int32
atomic.StoreInt64(&val, new)

:原子写入

int64

注意:所有Load和Store操作都必须传入变量地址。

示例:用原子操作控制程序运行状态

var running int32 = 1

func monitor() {for {if atomic.LoadInt32(&running) == 0 {fmt.Println("Stopping...")return}time.Sleep(100 * time.Millisecond)}}

func main() {go monitor()time.Sleep(2 time.Second)atomic.StoreInt32(&running, 0)time.Sleep(100 time.Millisecond)}

3. 比较并交换(Compare And Swap, CAS)

CAS是实现无锁算法的核心,只有当当前值等于旧值时,才将新值写入。

atomic.CompareAndSwapInt32(&val, old, new)
atomic.CompareAndSwapInt64(&val, old, new)
atomic.CompareAndSwapUint32(&val, old, new)
atomic.CompareAndSwapPointer(&ptr, old, new)

返回

bool

,表示是否交换成功。

示例:实现线程安全的单例初始化

var initialized int32var config *Config

func GetConfig() Config {if atomic.LoadInt32(&initialized) == 0 {atomic.CompareAndSwapInt32(&initialized, 0, 1)config = &Config{ / 初始化 */ }}return config}

注意:上面例子存在ABA问题风险,生产环境建议结合

sync.Once

或使用指针CAS更安全。

4. 交换操作(Swap)

原子地将新值写入变量,并返回旧值。

atomic.SwapInt32(&val, new)
atomic.SwapInt64(&val, new)
atomic.SwapPointer(&ptr, new)

示例:切换配置指针

var configPtr unsafe.Pointer

func updateConfig(newConfig *Config) {atomic.SwapPointer(&configPtr, unsafe.Pointer(newConfig))}

func getCurrentConfig() Config {return (Config)(atomic.LoadPointer(&configPtr))}

使用注意事项

原子操作只适用于基本类型,不能用于结构体整体(除非是

atomic.Value

)必须对变量地址操作,不能传值

int64

在32位系统上操作不是原子的,必须使用

atomic

包避免混合使用原子操作和普通读写,会导致数据竞争复杂逻辑建议使用

sync.Mutex

,原子操作适合轻量级场景

atomic.Value:任意类型的原子操作

Go还提供

atomic.Value

类型,可用于存储任意类型的值(需运行时确定),常用于配置热更新。

var config atomic.Value

func init() {config.Store(&Config{Version: "v1"})}

func updateConfig(newCfg *Config) {config.Store(newCfg)}

func getCurrent() Config {return config.Load().(Config)}

注意:

atomic.Value

一旦使用,就不能复制,且读写必须是相同类型。

基本上就这些。atomic包是Go并发编程中高效、底层的工具,掌握它能写出更轻量、高性能的并发代码。关键在于理解每种操作的语义和适用场景,避免误用导致竞态条件。不复杂但容易忽略细节。

以上就是Golang原子操作详解 atomic包函数使用的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1401312.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月15日 17:43:45
下一篇 2025年12月15日 17:43:57

相关推荐

  • Golang reflect反射机制 动态类型检查

    Go语言反射通过reflect包实现,可在运行时动态获取类型与值信息。使用reflect.TypeOf()和reflect.ValueOf()分别获取变量的类型和值,通过Type.Kind()和Value.Kind()判断底层类型,支持对结构体字段及标签的遍历与操作,常用于序列化、配置解析等场景。示…

    2025年12月15日
    000
  • Golang性能分析环境 pprof工具配置

    pprof能解决Go应用的CPU高占用、内存泄漏、协程阻塞、锁竞争等问题,通过在程序中导入”net/http/pprof”并启动HTTP服务,即可采集性能数据。使用时需注意生产环境安全,避免公网暴露,合理设置block和mutex采样率,区分heap与allocs内存分析,并…

    2025年12月15日
    000
  • Golang微服务错误处理 跨服务错误传递方案

    统一错误结构体设计(含Code、Message、Status等字段)实现跨服务错误传递,通过gRPC的google.rpc.Status扩展携带自定义错误详情,并在HTTP网关层映射为标准JSON响应,结合错误码集中管理与构造函数提升可维护性,确保多协议下错误信息一致可解析。 在Golang微服务架…

    2025年12月15日
    000
  • Golang桥接模式应用 抽象与实现解耦

    桥接模式通过接口与组合将抽象与实现分离,使设备与遥控器可独立扩展。定义Device接口并实现TV等具体设备,遥控器通过持有Device接口实现解耦,基础遥控器RemoteControl提供通用控制,高级遥控器AdvancedRemoteControl通过组合扩展功能,新增设备或遥控类型无需大量继承,…

    2025年12月15日
    000
  • Golang的hash哈希算法 MD5/SHA实现

    Go语言通过crypto包提供MD5、SHA系列哈希算法,适用于数据完整性校验;使用md5.New()、sha256.New()等创建哈希对象,配合io.WriteString或io.Copy处理字符串或文件;推荐SHA256以上算法以确保安全,避免MD5和SHA1用于敏感场景。 在Go语言中,标准…

    2025年12月15日
    000
  • Golang错误处理性能影响 对比异常与返回值开销

    Go语言通过返回值处理错误,避免了异常机制的栈展开开销,提升性能与可读性。错误作为普通返回值传递,无运行时负担,编译器可优化,CPU分支预测高效。相比Java、C++等语言的异常,Go的错误处理在正常与错误路径均更轻量,微基准测试显示性能高出一个数量级。该设计符合Go显式处理错误的哲学,适用于高并发…

    2025年12月15日
    000
  • Golang构建缓存优化 提升编译速度技巧

    Go语言构建缓存通过哈希校验源码、依赖、编译器版本等输入生成唯一标识,命中缓存时直接复用编译结果,避免重复编译,显著提升编译效率。 Go语言的构建缓存机制是提升编译速度的关键,它通过智能地重用之前编译过的包和模块,显著减少了重复工作。简单来说,就是Go编译器非常聪明,它会记住你之前编译过什么,如果发…

    2025年12月15日
    000
  • Golang并发模型核心是什么 解析GMP调度器工作原理

    GMP调度器是Go并发的核心,由G(协程)、M(线程)、P(逻辑处理器)构成,通过本地队列、工作窃取和抢占式调度实现高效并发,支持高并发性能。 Go语言的并发模型核心是“协程+通信”,即通过轻量级的协程(goroutine)和基于通道(channel)的通信机制来实现高效、安全的并发编程。这种设计避…

    2025年12月15日
    000
  • Golang WebSocket开发 实时通信服务搭建

    答案:基于Golang的WebSocket服务通过Hub管理连接与广播,利用Goroutine和Channel实现高效并发,结合readPump/writePump处理读写,send通道缓冲提供背压,定时ping/pong维持心跳,避免Goroutine泄漏与消息堆积,确保高可用与扩展性。 Gola…

    2025年12月15日
    000
  • Golang的select语句如何使用 讲解多路复用的实现原理

    在 go 语言中,select 是一种用于处理多个 channel 操作的控制结构,其核心作用是实现多路复用。1. 它类似于 switch,但专为 channel 设计;2. 支持同时监听多个 channel 的读写操作,并在任意一个就绪时处理;3. 若多个 channel 同时就绪,则随机选择一个…

    2025年12月15日 好文分享
    000
  • Golang指针与并发安全性 原子操作与互斥锁方案

    多个goroutine通过指针访问共享变量时可能引发数据竞争;2. Go通过原子操作和互斥锁实现并发安全;3. 指针直接操作内存需配合同步机制避免竞态条件。 在Go语言中,指针与并发编程经常一起出现,尤其是在多个goroutine共享数据时。如果不加控制地通过指针访问和修改共享变量,很容易引发数据竞…

    2025年12月15日
    000
  • GolangHTTP服务器搭建 路由处理实现

    Go语言通过net/http包实现HTTP服务器与路由处理,使用http.HandleFunc注册路由,http.ListenAndServe启动服务,默认基于DefaultServeMux;可通过自定义http.ServeMux实现更清晰的模块化路由管理。 在Go语言中搭建HTTP服务器并实现路由…

    2025年12月15日
    000
  • Golang反射修改值技巧 Value.Elem和Set方法

    要通过反射修改变量,必须传入指针并调用Elem()获取可设置的Value,再用Set或类型专用方法赋值,确保类型匹配且字段可导出。 在 Go 语言中,反射(reflect)是一种强大的机制,可以在运行时动态地查看和操作变量的值与类型。当我们需要通过反射修改一个变量的值时,必须确保该变量是可寻址且可设…

    2025年12月15日
    000
  • 怎样用反射实现通用函数 处理不同类型参数的技巧

    使用反射可动态处理不同类型参数,通过reflect.ValueOf获取值并判断类型,实现通用函数;2. 遍历结构体字段需判断Kind为Struct后,用NumField和Field遍历;3. 可通过反射调用方法,如调用结构体的Validate方法并判断返回值;4. 处理切片和映射时,用Kind判断后…

    2025年12月15日
    000
  • Golang微服务架构全景 完整解决方案

    答案:构建Golang微服务架构需围绕解耦、扩展、可用性与观测性,采用DDD划分服务,gRPC通信,Consul/Nacos做服务发现,Prometheus+Jaeger实现监控追踪,结合Kubernetes部署与Istio灰度发布,通过熔断限流提升韧性,统一API网关与JWT/mTLS保障安全,全…

    2025年12月15日
    000
  • Go程序处理HTTP大文件上传崩溃怎么优化

    go程序处理大文件上传崩溃的问题,通常是因为内存占用过高。解决方法是:1. 使用io.reader进行流式读取,避免一次性将整个文件加载到内存;2. 设置合适的缓冲区大小,通常在几kb到几mb之间;3. 使用multipart.reader逐个读取multipart/form-data中的part,…

    2025年12月15日 好文分享
    000
  • Golang全栈开发实践 前后端分离方案

    Golang可实现前后端分离全栈开发,后端用Gin等框架提供RESTful或GraphQL API,前端用React/Vue等框架构建界面,通过JSON交互,JWT实现认证,CORS处理跨域,Docker部署,发挥Golang高性能优势。 前后端分离,用Golang做全栈?当然可以!核心在于API的…

    2025年12月15日
    000
  • Golang处理图片的常用方法 使用imaging库裁剪缩放图片

    imaging库在golang图片处理中备受青睐,因为它提供了直观的api、优异的性能、全面的功能和活跃的社区支持,使得裁剪、缩放等高频操作更高效便捷,开发者无需关注底层细节即可快速实现图像处理任务。 Golang在图片处理方面,特别是面对裁剪和缩放这类高频操作时, imaging 库无疑是我的首选…

    2025年12月15日
    000
  • Golang测试覆盖率统计 go test -cover用法

    测试覆盖率反映测试对代码的覆盖程度,Go语言通过go test -cover命令生成覆盖率报告,结合-coverprofile可输出详细数据并生成HTML可视化报告,帮助识别未覆盖代码。报告中百分比表示已执行代码比例,但高覆盖率不等于高质量测试,需关注测试用例的全面性与边界、异常场景覆盖。提升覆盖率…

    2025年12月15日
    000
  • Golang指针声明方法 &和*操作符使用指南

    Go语言中指针通过&amp;amp;amp;取地址和*解引用操作实现对变量内存的访问,2. 指针可提升大对象处理和函数间数据共享的效率,3. 使用指针需注意nil判断以避免panic。 在Go语言中,指针是操作变量内存地址的重要工具。理解 &amp;amp;amp; 和 * 操作符的…

    2025年12月15日
    000

发表回复

登录后才能评论
关注微信