0

0

Go并发编程中atomic包如何使用_Go原子操作讲解

P粉602998670

P粉602998670

发布时间:2026-01-15 13:47:02

|

826人浏览过

|

来源于php中文网

原创

atomic.LoadUint64总返回0通常因传入未初始化或栈上临时变量的地址;须确保指针指向包级变量或堆分配的持久内存。

go并发编程中atomic包如何使用_go原子操作讲解

atomic.LoadUint64 为什么总返回 0?检查是否用了未初始化的指针

常见错误是把 uint64 变量地址传给 atomic.LoadUint64,但变量本身没初始化,或者传了上临时变量的地址。该函数只接受 *uint64,且要求地址有效、生命周期足够长。

  • 确保变量声明为包级变量或显式分配在堆上(如用 new(uint64)
  • 不要对局部变量取地址后传给原子操作函数,逃逸分析可能不保证安全
  • 确认没有重复声明同名变量覆盖了原始指针(比如循环里 var x uint64; p := &x
var counter uint64 = 100
// ✅ 正确:包级变量,地址稳定
fmt.Println(atomic.LoadUint64(&counter))

// ❌ 错误示例(编译能过,但行为未定义):
// func bad() {
//     var x uint64 = 42
//     fmt.Println(atomic.LoadUint64(&x)) // x 在函数返回后失效
// }

CompareAndSwapInt32 的失败不是 bug,而是并发控制的正常反馈

atomic.CompareAndSwapInt32 返回 bool 表示“旧值匹配且已更新”。它不抛异常,也不重试——这是设计使然。你得自己决定是否重试、何时放弃、要不要加 backoff。

  • 典型场景是实现无锁计数器、状态机跃迁(如从 0 → 1 表示“启动中”)
  • 如果期望“一定成功”,需配合 for 循环 + 条件判断,即 CAS 自旋
  • 注意:过度自旋会浪费 CPU,高竞争下建议结合 runtime.Gosched() 或短时休眠
var state int32 = 0
for !atomic.CompareAndSwapInt32(&state, 0, 1) {
    // 当前 state 不是 0,说明已被其他 goroutine 占用
    // 可选择继续等待、返回错误,或稍作让出
    runtime.Gosched()
}

StorePointer 和 LoadPointer 必须配对使用,且类型要严格一致

Go 的 atomic.StorePointeratomic.LoadPointer 操作的是 unsafe.Pointer,不进行类型检查。一旦存入和读出的底层结构不匹配(比如存了 *A 却当 *B 读),就会触发未定义行为,甚至崩溃。

LangChain
LangChain

一个开源框架,用于构建基于大型语言模型(LLM)的应用程序。

下载
  • 务必用 unsafe.Pointer(&x) 存,用 (*T)(atomic.LoadPointer(&p)) 读,且 T 必须和存入时的原始类型完全一致
  • 不能跨包直接传递裸 unsafe.Pointer,容易丢失类型上下文
  • Go 1.19+ 支持泛型封装,可写个类型安全的 wrapper,避免手写 unsafe
var ptr unsafe.Pointer
type Config struct{ Timeout int }
cfg := &Config{Timeout: 5}
atomic.StorePointer(&ptr, unsafe.Pointer(cfg))
// ✅ 正确读取
loaded := (*Config)(atomic.LoadPointer(&ptr))
fmt.Println(loaded.Timeout) // 5

不要用 atomic 替代 mutex 处理复杂状态,尤其涉及多个字段时

原子操作只保证单个值的读写线性化,无法原子地更新两个关联字段(如 countsum)。试图用多个 atomic 操作拼凑“逻辑原子性”,大概率导致中间态被其他 goroutine 观察到,引发数据不一致。

  • 当需要保护一组字段、或执行多步条件逻辑(如“若余额 > X 则扣款并记录日志”),必须用 sync.Mutexsync.RWMutex
  • atomic 适合高性能热点路径:计数器、开关标志、单指针替换、序列号生成等
  • 一个易忽略点:atomic 不提供内存屏障之外的同步语义,比如不会阻止编译器或 CPU 重排非原子访问

真正难的不是调用哪个函数,而是判断“这里到底需不需要原子性”以及“这个变量的修改是否独立于其他状态”。很多并发 bug 源于过早优化——先用 sync.Mutex 写正确,再看 profile 数据决定是否值得换成 atomic

相关文章

编程速学教程(入门课程)
编程速学教程(入门课程)

编程怎么学习?编程怎么入门?编程在哪学?编程怎么学才快?不用担心,这里为大家提供了编程速学教程(入门课程),有需要的小伙伴保存下载就能学习啦!

下载

本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

相关专题

更多
counta和count的区别
counta和count的区别

Count函数用于计算指定范围内数字的个数,而CountA函数用于计算指定范围内非空单元格的个数。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

197

2023.11.20

堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

388

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

571

2023.08.10

堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

388

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

571

2023.08.10

Java 桌面应用开发(JavaFX 实战)
Java 桌面应用开发(JavaFX 实战)

本专题系统讲解 Java 在桌面应用开发领域的实战应用,重点围绕 JavaFX 框架,涵盖界面布局、控件使用、事件处理、FXML、样式美化(CSS)、多线程与UI响应优化,以及桌面应用的打包与发布。通过完整示例项目,帮助学习者掌握 使用 Java 构建现代化、跨平台桌面应用程序的核心能力。

61

2026.01.14

php与html混编教程大全
php与html混编教程大全

本专题整合了php和html混编相关教程,阅读专题下面的文章了解更多详细内容。

31

2026.01.13

PHP 高性能
PHP 高性能

本专题整合了PHP高性能相关教程大全,阅读专题下面的文章了解更多详细内容。

72

2026.01.13

MySQL数据库报错常见问题及解决方法大全
MySQL数据库报错常见问题及解决方法大全

本专题整合了MySQL数据库报错常见问题及解决方法,阅读专题下面的文章了解更多详细内容。

20

2026.01.13

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Go 教程
Go 教程

共32课时 | 3.7万人学习

Go语言实战之 GraphQL
Go语言实战之 GraphQL

共10课时 | 0.8万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号