示例
通过如下的代码,我想通过开多个协程使得count累加10000次,值最终应该是10000,可是执行如下的代码发现值不等于10000,都是小于10000的值
func main(){
count := 0
for i:=0 ;i<10000;i++{
go func() {
count++
}()
}
time.Sleep(time.Second* 2)
fmt.Println(count)
}
结果
分析原因
造成该问题的原因是协程在读count值时,假设有a、b两个携程,a读取了count值为2,还未增加,这时b也读取了count值为2,a、b进行了加1,count=3.这种情况会在多个协程中发生,导致了最终count累加不到10000,相当于脏读。
这里我们可以使用Mutex来解决这个问题
Sync.Mutex
Sync.Mutex 可以进行加锁。在每个协程累加前读取Mutex,如果被锁则等待,这样就可以防止count在同一时间被重复读取导致最终数据出错,代码如下
func main(){
var m sync.Mutex
count := 0
for i:=0 ;i<10000;i++{
go func() {
m.Lock()
defer m.Unlock()
count++
}()
}
time.Sleep(time.Second* 2)
fmt.Println(count)
}
通过使用Mutex我们可以多次尝试,结果均为10000