说明
介绍
如何保证在一个goroutine中看到在另一个goroutine修改的变量的值,这篇文章进行了详细说明。
建议
如果程序中修改数据时有其他goroutine同时读取,那么必须将读取串行化。为了串行化访问,请使用channel或其他同步原语,例如sync和sync/atomic来保护数据。
先行发生
在一个gouroutine中,读和写一定是按照程序中的顺序执行的。即编译器和处理器只有在不会改变这个goroutine的行为时才可能修改读和写的执行顺序。由于重排,不同的goroutine可能会看到不同的执行顺序。例如,一个goroutine执行a = 1;b = 2;
,另一个goroutine可能看到b
在a
之前更新。
为了说明读和写的必要条件,我们定义了先行发生(Happens Before)
–Go程序中执行内存操作的偏序。如果事件e1
发生在e2
前,我们可以说e2
发生在e1
后。如果e1
不发生在e2
前也不发生在e2
后,我们就说e1
和e2
是并发的。
在单独的goroutine中先行发生的顺序即是程序中表达的顺序。
当下面条件满足时,对变量 v 的读操作 r 是 被允许 看到对 v 的写操作 w 的:
- r 不先行发生于 w
- 在 w 后 r 前没有对 v 的其他写操作
为了保证对变量 v 的读操作 r 看到对 v 的写操作 w ,要确保 w 是 r 允许看到的唯一写操作。即当下面条件满足时, r 被保证 看到 w :
- w 先行发生于 r
- 其他对共享变量 v 的写操作要么在 w 前,要么在 r 后。
这一对条件比前面的条件更严格,需要没有其他写操作与 w 或 r 并发发生。
单独的goroutine中没有并发,所以上面两个定义是相同的:读操作 r 看到最近一次的写操作 w 写入 v 的值。当多个goroutine访问共享变量 v 时,它们必须使用同步事件来建立先行发生这一条件来保证读操作能看到需要的写操作。
对变量 v 的零值初始化在内存模型中表现的与写操作相同。
对大于一个字的变量的读写操作表现的像以不确定顺序对多个一字大小的变量的操作。
同步
初始化
程序的初始化在单独的goroutine中进行,但这个goroutine可能会创建出并发执行的其他goroutine。
如果包p引入(import)包q,那么q的init函数的结束先行发生于p的所有init函数开始
main.main函数的开始发生在所有init函数结束之后
创建goroutine
go
关键字开启新的goroutine,先行发生于这个goroutine开始执行,例如下面程序:
1 | var a string |
调用hello
会在之后的某时刻打印出"hello, world"(可能在hello
返回之后)
销毁goroutine
gouroutine的退出并不会保证先行发生于程序的任何事件。例如下面程序:
1 | var a string |
没有用任何同步操作限制对a的赋值,所以并不能保证其他goroutine能看到a的变化。实际上,一个激进的编译器可能会删掉整个go语句。
如果想要在一个goroutine中看到另一个goroutine的执行效果,请使用锁或者channel这种同步机制来建立程序执行的相对顺序。
channel通信
channel通信是goroutine同步的主要方法。每一个在特定channel的发送操作都会匹配到通常在另一个goroutine执行的接收操作。
在channel的发送操作先行发生于对应的接收操作完成
例如:
1 | var c = make(chan int, 10) |
这个程序能保证打印出"hello, world"。对a的写先行发生于在c上的发送,先行发生于在c上的对应的接收完成,先行发生于print
。
对channel的关闭先行发生于接收到零值,因为channel已经被关闭了。
在上面的例子中,将c <- 0
替换为close(c)
还会产生同样的结果。
无缓冲channel的接收先行发生于发送完成
如下程序(和上面类似,只交换了对channel的读写位置并使用了非缓冲channel):
1 | var c = make(chan int) |
1 | func main() { |
此程序也能保证打印出"hello, world"。对a的写先行发生于从c接收,先行发生于向c发送完成,先行发生于print
。
如果是带缓冲的channel(例如c = make(chan int, 1)
),程序不保证打印出"hello, world"(可能打印空字符,程序崩溃或其他行为)。
在容量为C的channel上的第k个接收先行发生于从这个channel上的第k+C次发送完成。
这条规则将前面的规则推广到了带缓冲的channel上。可以通过带缓冲的channel来实现计数信号量:channel中的元素数量对应着活动的数量,channel的容量表示同时活动的最大数量,发送元素获取信号量,接收元素释放信号量,这是限制并发的通常用法。
下面程序为work
中的每一项开启一个goroutine,但这些goroutine通过有限制的channel来确保最多同时执行三个工作函数(w)。
1 | var limit = make(chan int, 3) |
锁
sync包实现了两个锁的数据类型sync.Mutex和sync.RWMutex。
对任意的sync.Mutex或sync.RWMutex变量l和n < m,n次调用l.Unlock()先行发生于m次l.Lock()返回
下面程序:
1 | var l sync.Mutex |
能保证打印出"hello, world"。第一次调用l.Unlock()(在f()中)先行发生于main中的第二次l.Lock()返回, 先行发生于print。
对于sync.RWMutex变量l,任意的函数调用l.RLock满足第n次l.RLock后发生于第n次调用l.Unlock,对应的l.RUnlock先行发生于第n+1次调用l.Lock。
Once
sync包的Once
为多个goroutine提供了安全的初始化机制。能在多个线程中执行once.Do(f)
,但只有一个f()
会执行,其他调用会一直阻塞直到f()
返回。
通过once.Do(f)
执行f()
先行发生(指f()返回)于其他的once.Do(f)
返回。
如下程序:
1 | var a string |
调用twoprint
会打印"hello, world"两次。setup
只在第一次doprint
时执行。
错误的同步方法
注意,读操作 r 可能会看到并发的写操作 w 。即使这样也不能表明 r 之后的读能看到 w 之前的写。
如下程序:
1 | var a, b int |
g
可能先打印出2然后是0。
这个事实证明一些旧的习惯是错误的。
双重检查锁定是为了避免同步的资源消耗。例如twoprint程序可能会错误的写成:
1 | var a string |
在doprint
中看到done
被赋值并不保证能看到对a
赋值。此程序可能会错误地输出空字符而不是"hello, world"。
另一个错误的习惯是忙等待 例如:
1 | var a string |
和之前程序类似,在main中看到done
被赋值不能保证看到a
被赋值,所以此程序也可能打印出空字符。更糟糕的是因为两个线程间没有同步事件,在main中可能永远不会看到done
被赋值,所以main中的循环不保证能结束。
对程序做一个微小的改变:
1 | type T struct { |
即使main看到了g != nil
并且退出了循环,也不能保证看到g.msg的初始化值。
在上面所有的例子中,解决办法都是相同的:明确的使用同步。