Golang并发程序的内存可见性说明
发布时间 - 2026-01-05 00:00:00 点击率:次Go中不存在独立的“内存可见性”概念,其并发问题本质是data race;应使用sync.Mutex或sync.RWMutex保护共享变量,或通过channel通信传递副本,避免共享内存;sync/atomic仅适用于基础类型原子操作,且无法保证多变量协同状态一致性。
Go 中没有“内存可见性”这个独立概念
Go 语言规范不定义类似 Java 的 happens-before 模型,也不提供 volatile 关键字。所谓“并发内存可见性问题”,在 Go 中本质是 data race —— 即多个 goroutine 同时读写同一变量且无同步约束。Go 的 go run -race 工具能检测它,但不会帮你“修复可见性”,只会报错。
用 sync.Mutex 或 sync.RWMutex 保护共享变量
这是最直接、最不容易出错的方案。只要所有读写都经由同一把锁保护,就天然满足顺序一致性语义,不存在“写完读不到”的情况。
- 不要只在写操作加锁、读操作裸奔 —— 这仍是 data race
- 避免锁粒度过大(如整个函数体加锁),但更不能为了性能漏锁关键字段
-
sync.RWMutex在读多写少场景下可提升吞吐,但注意RUnlock()必须配对,否则会死锁
var mu sync.RWMutex
var counter int
func Inc() {
mu.Lock()
counter++
mu
.Unlock()
}
func Get() int {
mu.RLock()
defer mu.RUnlock()
return counter
}
channel 是首选的通信方式,不是共享内存的替代品
Go 的哲学是 “Don’t communicate by sharing memory; share memory by communicating”。用 channel 传递值,本质上是复制而非共享,天然规避可见性与竞争问题。
- 向 channel 发送一个
int,接收方拿到的是副本,和发送方的原始变量无关 - 不要用 channel 传递指针并期望在接收端修改原变量 —— 这又回到了共享内存的老路
- 无缓冲 channel 的 send/receive 操作本身构成同步点,隐含 happens-before 关系
sync/atomic 仅适用于基础类型且需严格对齐
sync/atomic 提供原子读写,但仅支持 int32、int64、uint32、uint64、uintptr、unsafe.Pointer 和若干指针类型。它不保证其他字段的可见性,也不能用于 struct 字段级原子操作。
立即学习“go语言免费学习笔记(深入)”;
- 32 位系统上
int64原子操作要求 8 字节对齐,否则 panic(panic: runtime error: invalid memory address or nil pointer dereference) - 不要混用
atomic.StoreInt64()和普通赋值 —— 后者不提供原子性也不触发内存屏障 -
atomic.LoadUint64()读到的一定是某次atomic.StoreUint64()写入的完整值,但不保证“最新”—— 它只是原子,不是自动刷新缓存
真正容易被忽略的是:即使用了 atomic,若逻辑上依赖多个变量的协同状态(比如 status + data),单靠 atomic 无法保证它们的组合状态一致 —— 这时候必须回到 mutex 或重新设计数据流。
# java
# go
# golang
# app
# 字节
# 工具
# golang并发
相关栏目:
【
网站优化151355 】
【
网络推广146373 】
【
网络技术251813 】
【
AI营销90571 】
相关推荐:
如何快速生成凡客建站的专业级图册?
如何构建满足综合性能需求的优质建站方案?
香港服务器网站测试全流程:性能评估、SEO加载与移动适配优化
Laravel Sail是什么_基于Docker的Laravel本地开发环境Sail入门
如何在阿里云购买域名并搭建网站?
Laravel怎么实现一对多关联查询_Laravel Eloquent模型关系定义与预加载【实战】
阿里云网站搭建费用解析:服务器价格与建站成本优化指南
JavaScript中如何操作剪贴板_ClipboardAPI怎么用
简历在线制作网站免费版,如何创建个人简历?
头像制作网站在线观看,除了站酷,还有哪些比较好的设计网站?
宙斯浏览器视频悬浮窗怎么开启 边看视频边操作其他应用教程
Laravel如何记录日志_Laravel Logging系统配置与自定义日志通道
谷歌浏览器下载文件时中断怎么办 Google Chrome下载管理修复
Laravel如何安装Breeze扩展包_Laravel用户注册登录功能快速实现【流程】
Laravel怎么上传文件_Laravel图片上传及存储配置
谷歌Google入口永久地址_Google搜索引擎官网首页永久入口
如何在阿里云香港服务器快速搭建网站?
Laravel中的Facade(门面)到底是什么原理
大连企业网站制作公司,大连2025企业社保缴费网上缴费流程?
Laravel Facade的原理是什么_深入理解Laravel门面及其工作机制
laravel怎么为API路由添加签名中间件保护_laravel API路由签名中间件保护方法
进行网站优化必须要坚持的四大原则
Laravel Eloquent关联是什么_Laravel模型一对一与一对多关系精讲
在线制作视频的网站有哪些,电脑如何制作视频短片?
智能起名网站制作软件有哪些,制作logo的软件?
Laravel如何生成和使用数据填充?(Seeder和Factory示例)
Laravel Pest测试框架怎么用_从PHPUnit转向Pest的Laravel测试教程
Laravel如何与Vue.js集成_Laravel + Vue前后端分离项目搭建指南
如何批量查询域名的建站时间记录?
Laravel如何使用缓存系统提升性能_Laravel缓存驱动和应用优化方案
Laravel项目怎么部署到Linux_Laravel Nginx配置详解
如何快速搭建二级域名独立网站?
Laravel如何实现全文搜索_Laravel Scout集成Algolia或Meilisearch教程
Laravel如何使用Service Provider服务提供者_Laravel依赖注入与容器绑定【深度】
企业在线网站设计制作流程,想建设一个属于自己的企业网站,该如何去做?
网站制作软件免费下载安装,有哪些免费下载的软件网站?
Claude怎样写结构化提示词_Claude结构化提示词写法【教程】
Laravel如何实现API版本控制_Laravel API版本化路由设计策略
手机软键盘弹出时影响布局的解决方法
linux写shell需要注意的问题(必看)
油猴 教程,油猴搜脚本为什么会网页无法显示?
UC浏览器如何设置启动页 UC浏览器启动页设置方法
如何快速登录WAP自助建站平台?
大连网站制作费用,大连新青年网站,五年四班里的视频怎样下载啊?
制作ppt免费网站有哪些,有哪些比较好的ppt模板下载网站?
无锡营销型网站制作公司,无锡网选车牌流程?
活动邀请函制作网站有哪些,活动邀请函文案?
JavaScript Ajax实现异步通信
laravel怎么为应用开启和关闭维护模式_laravel应用维护模式开启与关闭方法
Laravel如何实现多对多模型关联?(Eloquent教程)


.Unlock()
}
func Get() int {
mu.RLock()
defer mu.RUnlock()
return counter
}