go中常用的同步原语<二>Mutex

本文通过示例深入解析了Go语言中Mutex互斥锁的使用,展示了如何解决多goroutine并发操作共享资源时的数据竞态问题。通过对比无锁和加锁的代码,突出了Mutex在保证线程安全方面的重要性。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

go中常用的同步原语 <二> Mutex

有时候在Go代码中可能会存在多个goroutine同时操作一个资源(临界区),这种情况会发生竞态问题(数据竞态)。类比现实生活中的例子有十字路口被各个方向的的汽车竞争;还有火车上的卫生间被车厢里的人竞争。

//在没有锁的情况下
package main
import (
	"fmt"
	"time"
)
var n = 0
func count() {
	fmt.Println(n)
	n = n + 1
}
func main() {
	go count()
	go count()
	time.Sleep(1 * time.Hour)   
	//代码执行到这来 n的值是不确定的因为开启多个goroutine后 有线程安全问题
	//n的值有可能是2 ,也有可能是1
	//为了解决这个问题就需要我们今天的主角 Mutex 互斥锁了
}

互斥锁是一种常用的控制共享资源访问的方法,它能够保证同时只有一个goroutine可以访问共享资源。Go语言中使用sync包的Mutex类型来实现互斥锁。
使用互斥锁能够保证同一时间有且只有一个goroutine进入临界区,其他的goroutine则在等待锁;当互斥锁释放后,等待的goroutine才可以获取锁进入临界区,多个goroutine同时等待一个锁时,唤醒的策略是随机的。

使用互斥锁的代码如下

package main
import (
	"fmt"
	"sync"
	"time"
)
var n = 0
var mu = sync.Mutex{}
func count() {
	mu.Lock() //进入方法之后立刻上锁 防止其他goroutine拿到n的值
	defer mu.Unlock()  //执行完方法 开锁把操作权让给其他goroutine
	fmt.Println(n)
	n = n + 1
}
func main() {
	go count()
	go count()
	time.Sleep(1 * time.Hour)  //睡眠一段时间等待goroutine执行完毕
	//这里的n就一定会为2
}

引用:

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值