0

0

如何在Golang中减少锁竞争_Golang sync锁与原子操作优化示例

P粉602998670

P粉602998670

发布时间:2026-01-02 18:41:05

|

855人浏览过

|

来源于php中文网

原创

sync.Mutex在高并发下易成瓶颈,因其排他性导致读操作串行化、调度开销上升;RWMutex仅适用于读多写少且读操作真正只读的场景;atomic适用于基础类型单变量操作,性能更高但功能受限;锁粒度细化(如分片锁)可显著提升QPS,但需权衡分片数与资源消耗。

如何在golang中减少锁竞争_golang sync锁与原子操作优化示例

为什么 sync.Mutex 在高并发下容易成为瓶颈

因为 sync.Mutex 是排他锁,一旦某个 goroutine 持有锁,其他所有 goroutine 都得排队等待。在读多写少场景(比如计数器、配置缓存)中,读操作被强制串行化,吞吐量直接掉一半以上。Go 运行时的锁唤醒调度开销也会随竞争加剧而上升,runtime.futex 调用频繁出现就是信号。

sync.RWMutex 替代 sync.Mutex 的前提和陷阱

sync.RWMutex 只在读多写少且写操作不频繁时才有效。它允许多个 goroutine 同时读,但写会阻塞所有读和写。注意:如果写操作太频繁,或者读操作里隐含了写(比如读完立刻修改字段),RWMutex 不但没收益,反而因额外的锁状态管理更慢。

  • 必须确保读操作是真正只读的——不能在 RWMutex.RLock() 保护的代码块里调用任何可能修改共享数据的方法
  • RLock()RUnlock() 必须成对出现;漏掉 RUnlock() 会导致后续所有写操作永久阻塞
  • 不要在 defer 中无条件调用 RUnlock(),因为 RLock() 失败(如被 context 取消)时不该解锁
var mu sync.RWMutex
var config map[string]string

func Get(key string) string {
    mu.RLock()
    defer mu.RUnlock() // ✅ 安全:只要 RLock 成功,就一定需要 RUnlock
    return config[key]
}

func Set(key, value string) {
    mu.Lock()
    defer mu.Unlock()
    config[key] = value
}

什么时候该用 atomic 而不是锁

atomic 包只适用于基础类型(int32int64uint32uintptr*Tunsafe.Pointer)的单变量读写。它比锁快一个数量级,且无 goroutine 阻塞。但无法用于结构体字段更新、多字段协调、或需要条件判断再写入的逻辑(比如“如果值为 0 才设为 1”需用 atomic.CompareAndSwapInt32)。

阿里云-虚拟数字人
阿里云-虚拟数字人

阿里云-虚拟数字人是什么? ...

下载
  • atomic.LoadInt64(&x)atomic.StoreInt64(&x, v) 是最常用、最安全的组合
  • 避免对同一变量混用 atomic 和非原子操作(如直接赋值 x = 1),这会破坏内存可见性保证
  • atomic.Value 可安全存储任意类型指针(如 *Config),适合替换整个只读结构体,但每次 Store 都分配新对象,别在热路径反复 Store
var counter int64

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

func GetCount() int64 {
    return atomic.LoadInt64(&counter)
}

锁粒度细化:从全局锁到分片锁的实际取舍

当无法避免锁,又存在明显热点(如哈希表某 bucket 冲突严重),可把一个大锁拆成多个小锁。典型做法是按 key 哈希后对 N 取模,映射到 [N]*sync.Mutex 数组。N 一般取 32 或 64——太小仍竞争,太大内存浪费且 cache line false sharing 风险上升。

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

  • 分片数不宜动态调整,否则 rehash 期间需同时持有所有锁,反而更卡
  • 不要用 map[int]*sync.Mutex 存锁,遍历或 GC 时可能触发锁竞争;固定大小数组更可控
  • 若业务允许弱一致性(如统计误差可接受),甚至可用 atomic + 分片计数器,完全避开锁
const shardCount = 64
var shards [shardCount]struct {
    mu sync.Mutex
    m  map[string]int
}

func Get(key string) int {
    idx := int(uintptr(unsafe.Pointer(&key)) % shardCount)
    shards[idx].mu.Lock()
    defer shards[idx].mu.Unlock()
    return shards[idx].m[key]
}
实际压测中,从全局 sync.Mutex 切到分片锁,QPS 提升常达 3–5 倍;但若分片数超过 CPU core 数太多,调度开销反而盖过收益。原子操作看似简单,但 atomic.ValueStore 是复制语义,大对象要小心 GC 压力。

相关专题

更多
golang如何定义变量
golang如何定义变量

golang定义变量的方法:1、声明变量并赋予初始值“var age int =值”;2、声明变量但不赋初始值“var age int”;3、使用短变量声明“age :=值”等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

174

2024.02.23

golang有哪些数据转换方法
golang有哪些数据转换方法

golang数据转换方法:1、类型转换操作符;2、类型断言;3、字符串和数字之间的转换;4、JSON序列化和反序列化;5、使用标准库进行数据转换;6、使用第三方库进行数据转换;7、自定义数据转换函数。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

225

2024.02.23

golang常用库有哪些
golang常用库有哪些

golang常用库有:1、标准库;2、字符串处理库;3、网络库;4、加密库;5、压缩库;6、xml和json解析库;7、日期和时间库;8、数据库操作库;9、文件操作库;10、图像处理库。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

335

2024.02.23

golang和python的区别是什么
golang和python的区别是什么

golang和python的区别是:1、golang是一种编译型语言,而python是一种解释型语言;2、golang天生支持并发编程,而python对并发与并行的支持相对较弱等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

206

2024.03.05

golang是免费的吗
golang是免费的吗

golang是免费的。golang是google开发的一种静态强类型、编译型、并发型,并具有垃圾回收功能的开源编程语言,采用bsd开源协议。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

388

2024.05.21

golang结构体相关大全
golang结构体相关大全

本专题整合了golang结构体相关大全,想了解更多内容,请阅读专题下面的文章。

194

2025.06.09

golang相关判断方法
golang相关判断方法

本专题整合了golang相关判断方法,想了解更详细的相关内容,请阅读下面的文章。

189

2025.06.10

golang数组使用方法
golang数组使用方法

本专题整合了golang数组用法,想了解更多的相关内容,请阅读专题下面的文章。

191

2025.06.17

php源码安装教程大全
php源码安装教程大全

本专题整合了php源码安装教程,阅读专题下面的文章了解更多详细内容。

150

2025.12.31

热门下载

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

精品课程

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

共32课时 | 3.2万人学习

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号