0

0

如何在Golang中实现自旋锁SpinLock Go语言低延迟锁实现

P粉602998670

P粉602998670

发布时间:2026-03-11 15:15:32

|

837人浏览过

|

来源于php中文网

原创

sync.mutex 不适合超低延迟场景,因其争用时触发 futex 系统调用进入内核态,上下文切换开销远超 100ns;自旋锁则全程用户态忙等,适合极短临界区。

如何在golang中实现自旋锁spinlock go语言低延迟锁实现

为什么 sync.Mutex 不适合超低延迟场景

因为 sync.Mutex 在争用时会触发系统调用(如 futex),进入内核态,哪怕只等几纳秒,上下文切换开销也远超 100ns;而自旋锁全程在用户态忙等,适合临界区极短(

常见错误现象:goroutine 频繁阻塞在 Lock() 上,pprof 显示大量时间花在 runtime.futex 或调度器等待——这不是锁写得不好,是选错了锁类型。

  • 仅当临界区是纯计算、无 I/O、无 channel 操作、无函数调用(或仅内联小函数)时才考虑自旋
  • 必须配合 GOMAXPROCS ≥ 实际物理核心数,否则自旋线程可能被抢占,反而更慢
  • Go 1.19+ 的 runtime.Spinning 状态可被 pprof 观察,但不会暴露给用户代码

手写 SpinLock 的最小安全实现

Go 没有原子布尔的“测试并置位”(TAS)内置指令封装,得靠 atomic.CompareAndSwapInt32 模拟;同时必须插入 runtime.Gosched() 防止单核死锁,并用 runtime.Pause()(Go 1.22+)或 runtime.nanosleep()(旧版)降低功耗。

type SpinLock struct {
    state int32 // 0 = unlocked, 1 = locked
}

func (s *SpinLock) Lock() {
    for !atomic.CompareAndSwapInt32(&s.state, 0, 1) {
        runtime.Pause() // Go 1.22+, 否则用 runtime.nanosleep(1)
    }
}

func (s *SpinLock) Unlock() {
    atomic.StoreInt32(&s.state, 0)
}
  • 不能用 atomic.AddInt32 替代 CAS,会导致 ABA 问题和丢失唤醒
  • Unlock() 必须用 StoreInt32,不能用 CAS,否则多 unlock 会 panic
  • 禁止在 Lock() 中加 time.Sleep 或任何非内联系统调用,那已不是自旋锁

比手写更靠谱的选择:sync/atomic + 自定义逻辑

多数所谓“低延迟需求”,其实只是想避免 mutex 的调度开销,但又不敢承担纯自旋风险——这时应放弃封装成锁,直接用原子操作+业务逻辑合并临界区。

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

IBM Watson
IBM Watson

IBM Watson文字转语音

下载

例如计数器累加:不用锁,改用 atomic.AddInt64(&counter, 1);状态机转换:用 atomic.CompareAndSwapUint32 做状态跃迁校验。

  • 只要操作本身是原子的(读-改-写可单条指令完成),就根本不需要锁
  • 多个字段需同步更新?把它们打包进一个 struct,用 atomic.Value 替代(注意分配逃逸)
  • 真要锁保护复杂结构,优先考虑 sync.RWMutex 读优化,而非自旋

调试和压测时最容易忽略的三个点

自旋锁的问题往往在压测后期才暴露,且难以复现。

  • 没有绑定 CPU 核心时,runtime.Gosched() 可能让 goroutine 迁移到其他 NUMA 节点,导致缓存行失效,实际延迟翻倍
  • Go 的 GC STW 期间,自旋 goroutine 仍会空转,可能拖慢整个 STW,让 GC 时间变长(观察 gctrace 中的 STW 字段)
  • go tool trace 查看 Proc Status 时,持续显示 Running 但无用户代码执行,大概率是自旋未退避或 Pause 失效

真正难的不是写对那十几行,是判断此刻该不该用它——临界区够不够短、负载是否稳定、机器有没有超线程干扰,这些都比锁本身更影响结果。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

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

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

210

2024.02.23

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

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

247

2024.02.23

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

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

356

2024.02.23

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

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

214

2024.03.05

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

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

409

2024.05.21

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

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

490

2025.06.09

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

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

200

2025.06.10

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

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

1438

2025.06.17

C# ASP.NET Core微服务架构与API网关实践
C# ASP.NET Core微服务架构与API网关实践

本专题围绕 C# 在现代后端架构中的微服务实践展开,系统讲解基于 ASP.NET Core 构建可扩展服务体系的核心方法。内容涵盖服务拆分策略、RESTful API 设计、服务间通信、API 网关统一入口管理以及服务治理机制。通过真实项目案例,帮助开发者掌握构建高可用微服务系统的关键技术,提高系统的可扩展性与维护效率。

3

2026.03.11

热门下载

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

精品课程

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

共32课时 | 6.1万人学习

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

共10课时 | 0.9万人学习

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

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