当前位置: 首页 > ai >正文

Go语言实战案例:使用sync.Mutex实现资源加锁

在并发编程中,数据共享是一把双刃剑。如果多个协程对同一个资源进行读写而没有任何同步机制,就可能会出现“竞态条件”或“数据竞争”的问题。Go语言为我们提供了 sync.Mutex,一种最基础也是最常用的加锁方式,用于保证在任意时刻只有一个 goroutine 能访问共享资源。

一、什么是 Mutex?

Mutex 是 mutual exclusion(互斥)的缩写,是 Go 标准库 sync 包提供的一种锁机制。它确保同一时间只有一个 goroutine 能进入临界区访问共享资源。

type Mutex struct {// 内部实现省略
}

常用方法:

  • • Lock():获取锁,如果锁已被占用,则阻塞等待;
  • • Unlock():释放锁,其他阻塞的 goroutine 才能继续执行。

二、为什么需要加锁?

设想一个并发场景:多个 goroutine 同时对一个整数进行自增操作。虽然操作看似简单,但由于 i++ 并不是原子操作,它会被分解为三个步骤:

  1. 1. 加载变量值;
  2. 2. 执行加一;
  3. 3. 保存回变量。

在这个过程中,如果没有互斥机制,多个 goroutine 很容易互相干扰,造成结果错误。

三、实战案例:并发安全的计数器

我们将构建一个并发安全的计数器,多个 goroutine 同时对它执行自增操作,确保最终计数正确。

1. 未加锁示例(存在竞态)

package mainimport ("fmt""sync"
)var counter intfunc main() {var wg sync.WaitGroupfor i := 0; i < 1000; i++ {wg.Add(1)go func() {counter++ // 非线程安全wg.Done()}()}wg.Wait()fmt.Println("最终计数器结果:", counter)
}

多次运行你会发现,结果每次都不一样,且通常小于1000。说明有些操作被“丢失”了。

2. 使用 sync.Mutex 加锁

package mainimport ("fmt""sync"
)var (counter intmu      sync.Mutex
)func main() {var wg sync.WaitGroupfor i := 0; i < 1000; i++ {wg.Add(1)go func() {mu.Lock()counter++ // 临界区mu.Unlock()wg.Done()}()}wg.Wait()fmt.Println("最终计数器结果:", counter)
}

每次运行的结果都是 1000,说明加锁成功避免了竞态条件。

四、延伸:封装一个线程安全的计数器结构

为了更好地管理共享资源,我们可以将计数器封装成一个结构体,并内置锁机制。

type SafeCounter struct {mu sync.Mutexval int
}func (s *SafeCounter) Inc() {s.mu.Lock()s.val++s.mu.Unlock()
}func (s *SafeCounter) Value() int {s.mu.Lock()defer s.mu.Unlock()return s.val
}

使用方式:

func main() {var wg sync.WaitGroupcounter := SafeCounter{}for i := 0; i < 1000; i++ {wg.Add(1)go func() {counter.Inc()wg.Done()}()}wg.Wait()fmt.Println("最终计数器结果:", counter.Value())
}

五、sync.Mutex 使用建议

✅ 适用场景

  • • 多个 goroutine 对同一资源进行读写时;
  • • 逻辑简单,读写比例相近的场景。

⚠️ 注意事项

  • • 死锁风险:锁住资源后若忘记释放会造成程序阻塞;
  • • 性能瓶颈:锁会阻塞其他协程,过多使用可能降低并发效率;
  • • 应尽量缩小锁的范围:只包裹必要的临界区。

六、与其他同步机制对比

同步方式特点
sync.Mutex最基础的锁,适用于简单同步控制
sync.RWMutex读写锁,适用于读多写少的场景
channel通过通信代替共享数据,较为优雅
sync.Atomic 操作支持原子操作,性能比 mutex 高

七、结语

通过本案例我们深入理解了 Go 中 sync.Mutex 的使用方式和适用场景。虽然它使用起来非常简单,但如果忽略其潜在的陷阱,可能会导致难以发现的 bug 和性能问题。掌握并合理使用 Mutex 是每一位 Go 开发者进行并发编程的第一步。

http://www.xdnf.cn/news/17053.html

相关文章:

  • 解决docker load加载tar镜像报json no such file or directory的错误
  • 安科瑞智慧能源管理系统在啤酒厂5MW分布式光伏防逆流控制实践
  • Redis协议数据迁移方式
  • 基于华为开发者空间的Open WebUI数据分析与可视化实战
  • 一次“无告警”的服务器宕机分析:从无迹可寻到精准定位
  • Java技术栈/面试题合集(8)-Redis篇
  • linux服务器上word转pdf后乱码问题
  • 如何对云环境或者超融合系统进行性能测试?
  • Java项目:基于SSM框架实现的公益网站管理系统【ssm+B/S架构+源码+数据库+毕业论文+答辩PPT+远程部署】
  • Python 3.13 预览版:颠覆性特性与实战指南
  • Spring 的依赖注入DI是什么?
  • Starrocks中的 Query Profile以及explain analyze及trace命令中的区别
  • 力扣经典算法篇-43-全排列(经典回溯问题)
  • SpringCloud学习------Gateway详解
  • 数据结构 | 树的秘密
  • WPF 与 Winform :Windows 桌面开发该用谁?
  • 剖析 DC - DC 电路 SW 节点铺铜面积大小的利弊
  • Redis集群模式下确保Key在同一Slot的实现方法
  • 依托CGA匹兹堡睡眠质量指数,优化老年睡眠干预方案​
  • 【面试场景题】日志去重与统计系统设计
  • 复制网页文字到Word、WPS文字?选中后直接拖放
  • PowerShell 入门2: 使用帮助系统
  • Apifox使用mock模仿后端返回数据
  • 基于ZYNQ ARM+FPGA的声呐数据采集系统设计
  • Go语言 定时器
  • 数据结构---Makefile 文件(格式、文件变量、调用、伪目标)、gcc编译的四个步骤、双向链表(概念、作用、应用)
  • Android 之 Kotlin中的kapt
  • 风丘助力混合动力汽车工况测试:精准采集整车信号解决方案
  • 【Spring Cloud】-- 注册中心
  • K8S的NetworkPolicy使用教程