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

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
}

引用:


版权声明:本文为jcetpoor原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接和本声明。