今天聊一聊Golang的互斥锁吧

2021-09-16 10:50:08 浏览数 (1)

今天聊一聊Golang的互斥锁吧

今天我们来聊一聊锁吧,我们都知道有并发就有并发安全的问题。对于有的变量不能是并发运行访问的。比如银行的存取款业务,假如可以并发进行的话,你想一想你往银行存这个月的工资200万,你老婆同一时间在银行取200万去做美容。假如不使用锁,你存完之后发现金额没有变化,你老婆取完钱后发现钱也没有变化。你是慌死了,那你老婆不高兴坏了…

所以我们这里就需要用到锁,当一个人访问这个业务时,就给它加上锁,别人就不能访问了。

看一看这个存钱的例子:

代码语言:javascript复制
var wg sync.WaitGroup
func main() {
	var money = 2000
	for i:=0;i<10;i  {
		wg.Add(1)
		go func() {
			for j:=0;j<100;j  {
				money  = 1
			}
			wg.Done()
		}()
	}
	wg.Wait()
	fmt.Println("最终金额",money)
}

这个例子就是10个人每个人给你存100块钱。这一百块钱分一百次存。这样存完后我们就有三千块钱了。

我们看一看运行结果:

代码语言:javascript复制
最终金额 3000

好像是没问题哦!那我们加大一下存款金额吧。让10个人每个人存1000,这一千块钱分一千次存,这样我们就会得到一万二千块钱,来看一看运行结果吧!

代码语言:javascript复制
最终金额 10366

是不是和我们预想得不一样?

这就是出现了并发安全问题。

对于这种问题,我们应该不允许并发访问。

然后我们看看怎么使用互斥锁解决这类问题吧!

代码语言:javascript复制
func main() {
	var money = 2000

	var mt sync.Mutex

	wg.Add(1)
	go func() {
		fmt.Println("搏达试图抢断")

		mt.Lock()
		fmt.Println("搏达抢断成功")

		money -= 300
		<- time.After(10 * time.Second)

		mt.Unlock()
		fmt.Println("搏达扔了球")
		wg.Done()
	}()

	wg.Add(1)
	go func() {
		fmt.Println("搏达试图跳舞")

		mt.Lock()
		fmt.Println("搏达跳舞成功")

		money -= 500
		<- time.After(10 * time.Second)
		mt.Unlock()
		fmt.Println("搏达放弃跳舞")
		wg.Done()
	}()

	wg.Wait()

}

这段程序的意义是两个协程同时抢锁,跳舞协程先抢到锁的话,搏达就开始跳舞,然后跳完舞解锁,抢断协程开始抢到锁,然后搏达结束跳舞开始抢断。如果抢断协程先抢到锁的话,搏达就先开始抢断然后再跳舞。

运行结果是

代码语言:javascript复制
搏达试图抢断
搏达抢断成功
搏达试图跳舞
搏达扔了球
搏达跳舞成功
搏达放弃跳舞

我们可以看到,搏达扔了球才能开始跳舞。这就是锁的功劳,让搏达不至于一边跳舞一边抢断而累趴。

0 人点赞