Go 内存模型 并发可见性

开发 前端
Go内存模型指定了在何种条件下可以保证在一个 goroutine 中读取变量时观察到不同 goroutine 中写入该变量的值。

[[409588]]

TLTR

  • 协程之间的数据可见性满足HappensBefore法则,并具有传递性
  • 如果包 p 导入包 q,则 q 的 init 函数的完成发生在任何 p 的操作开始之前
  • main.main 函数的启动发生在所有 init 函数完成之后
  • go 语句启动新的协程发生在新协程启动开始之前
  • go 协程的退出并不保证发生在任何事件之前
  • channel 上的发送发生在对应 channel 接收之前
  • 无buffer channel 的接收发生在发送操作完成之前
  • 对于容量为C的buffer channel来说,第k次从channel中接收,发生在第 k + C 次发送完成之前。
  • 对于任何的 sync.Mutex 或者 sync.RWMutex 变量 ,且有 n<m ,第 n 个调用 UnLock 一定发生在 m  Lock`之前。
  • 从 once.Do(f) 对 f() 的单个调用返回在任何一个 once.Do(f) 返回之前。
  • 如果两个动作不满足HappensBefore,则顺序无法预测

介绍

Go内存模型指定了在何种条件下可以保证在一个 goroutine 中读取变量时观察到不同 goroutine 中写入该变量的值。

建议

通过多个协程并发修改数据的程序必须将操作序列化。为了序列化访问,通过channel操作或者其他同步原语( sync 、 sync/atomic )来保护数据。

如果你必须要阅读本文的其他部分才能理解你程序的行为,请尽量不要这样...

Happens Before

在单个 goroutine 中,读取和写入的行为必须像按照程序指定的顺序执行一样。 也就是说,只有当重新排序不会改变语言规范定义的 goroutine 中的行为时,编译器和处理器才可以重新排序在单个 goroutine 中执行的读取和写入。 由于这种重新排序,一个 goroutine 观察到的执行顺序可能与另一个 goroutine 感知的顺序不同。 例如,如果一个 goroutine 执行 a = 1; b = 2;,另一个可能会在 a 的更新值之前观察到 b 的更新值。

为了满足读写的需求,我们定义了 happens before ,Go程序中内存操作的局部顺序。如果事件 e1 在 e2 之前发生,我们说 e2 在 e1 之后发生。还有,如果 e1 不在 e2 之前发生、 e2 也不在 e1 之前发生,那么我们说 e1 和 e2 并发happen。

在单个 goroutine 中, happens-before 顺序由程序指定。

当下面两个条件满足时,变量 v 的阅读操作 r 就 可能 观察到写入操作 w

  • r 不在 w 之前发生
  • 没有其他的请求 w2 发生在 w 之后, r 之前

为了保证 r 一定能阅读到 v ,保证 w 是 r 能观测到的唯一的写操作。当下面两个条件满足时, r 保证可以读取到 w

  • w 在 r 之前发生
  • 任何其他对共享变量 v 的操作,要么在 w 之前发生,要么在 r 之后发生

这一对条件比上一对条件更强;这要求无论是 w 还是 r ,都没有相应的并发操作。

在单个 goroutine 中,没有并发。所以这两个定义等价:读操作 r 能读到最近一次 w 写入 v 的值。但是当多个 goroutine 访问共享变量时,它们必须使用同步事件来建立 happens-before 关系。

使用变量 v 类型的0值初始化变量 v 的行为类似于内存模型中的写入。

对于大于单个机器字长的值的读取和写入表现为未指定顺序的对多个机器字长的操作。

同步

初始化

程序初始化在单个 goroutine 中运行,但该 goroutine 可能会创建其他并发运行的 goroutine。

如果包 p 导入包 q,则 q 的 init 函数的完成发生在任何 p 的操作开始之前。

main.main 函数的启动发生在所有 init 函数完成之后。

Go协程的创建

go 语句启动新的协程发生在新协程启动开始之前。

举个例子

  1. var a string 
  2.  
  3. func f() { 
  4.     print(a) 
  5.  
  6. func hello() { 
  7.     a = "hello, world" 
  8.     go f() 

调用 hello 将会打印 hello, world 。当然,这个时候 hello 可能已经返回了。

Go协程的销毁

go 协程的退出并不保证发生在任何事件之前

  1. var a string 
  2.  
  3. func hello() { 
  4.     go func() { a = "hello" }() 
  5.     print(a) 

对 a 的赋值之后没有任何同步事件,因此不能保证任何其他 goroutine 都会观察到它。 事实上,激进的编译器可能会删除整个 go 语句。

如果一个 goroutine 的效果必须被另一个 goroutine 观察到,请使用同步机制,例如锁或通道通信来建立相对顺序。

通道通信

通道通信是在go协程之间传输数据的主要手段。在特定通道上的发送总有一个对应的channel的接收,通常是在另外一个协程。

channel 上的发送发生在对应 channel 接收之前

  1. var c = make(chan int10
  2. var a string 
  3.  
  4. func f() { 
  5.     a = "hello, world" 
  6.     c <- 0 
  7.  
  8. func main() { 
  9.     go f() 
  10.     <-c 
  11.     print(a) 

程序能保证输出 hello, world 。对a的写入发生在往 c 发送数据之前,往 c 发送数据又发生在从 c 接收数据之前,它又发生在 print 之前。

channel 的关闭发生在从 channel 中获取到0值之前

在之前的例子中,将 c<-0 替换为 close(c) ,程序还是能保证输出 hello, world

无buffer channel 的接收发生在发送操作完成之前

这个程序,和之前一样,但是调换发送和接收操作,并且使用无buffer的channel

  1. var c = make(chan int
  2. var a string 
  3.  
  4. func f() { 
  5.     a = "hello, world" 
  6.     <-c 
  7.  
  8. func main() { 
  9.     go f() 
  10.     c <- 0 
  11.     print(a) 

也保证能够输出 hello, world 。对a的写入发生在c的接收之前,继而发生在c的写入操作完成之前,继而发生在print之前。

如果该 channel 是buffer channel (例如: c=make(chan int, 1) ),那么程序就不能保证输出 hello, world 。可能会打印空字符串、崩溃等等。从而,我们得到一个相对通用的推论:

对于容量为C的buffer channel来说,第k次从channel中接收,发生在第 k + C 次发送完成之前。

此规则将先前的规则推广到缓冲通道。 它允许通过buffer channel 来模拟信号量:通道中的条数对应活跃的数量,通道的容量对应于最大并发数。向channel发送数据相当于获取信号量,从channel中接收数据相当于释放信号量。 这是限制并发的常用习惯用法。

该程序为工作列表中的每个条目启动一个 goroutine,但是 goroutine 使用 limit channel进行协调,以确保一次最多三个work函数正在运行。

  1. var limit = make(chan int3
  2.  
  3. func main() { 
  4.     for _, w := range work { 
  5.         go func(w func()) { 
  6.             limit <- 1 
  7.             w() 
  8.             <-limit 
  9.         }(w) 
  10.     } 
  11.     select{} 

sync 包中实现了两种锁类型: sync.Mutex 和 sync.RWMutex

对于任何的 sync.Mutex 或者 sync.RWMutex 变量 ,且有 n<m ,第 n 个调用 UnLock 一定发生在 m  Lock`之前。

  1. var l sync.Mutex 
  2. var a string 
  3.  
  4. func f() { 
  5.     a = "hello, world" 
  6.     l.Unlock() 
  7.  
  8. func main() { 
  9.     l.Lock() 
  10.     go f() 
  11.     l.Lock() 
  12.     print(a) 

这个程序也保证输出 hello,world 。第一次调用 unLock 一定发生在第二次 Lock 调用之前

对于任何 sync.RWMutex 的 RLock 方法调用,存在变量n,满足 RLock 方法发生在第 n 个 UnLock 调用之后,并且对应的 RUnlock 发生在第 n+1 个 Lock 方法之前。

Once

在存在多个 goroutine 时, sync 包通过 once 提供了一种安全的初始化机制。对于特定的 f ,多个线程可以执行 once.Do(f) ,但是只有一个会运行 f() ,另一个调用会阻塞,直到 f() 返回

从 once.Do(f) 对 f() 的单个调用返回在任何一个 once.Do(f) 返回之前。

  1. var a string 
  2. var once sync.Once 
  3.  
  4. func setup() { 
  5.     a = "hello, world" 
  6.  
  7. func doprint() { 
  8.     once.Do(setup) 
  9.     print(a) 
  10.  
  11. func twoprint() { 
  12.     go doprint() 
  13.     go doprint() 

调用 twoprint 将只调用一次 setup。 setup 函数将在任一打印调用之前完成。 结果将是 hello, world 打印两次。

不正确的同步

注意,读取 r 有可能观察到了由写入 w 并发写入的值。尽管观察到了这个值,也并不意味着 r 后续的读取可以读取到 w 之前的写入。

  1. var a, b int 
  2.  
  3. func f() { 
  4.     a = 1 
  5.     b = 2 
  6.  
  7. func g() { 
  8.     print(b) 
  9.     print(a) 
  10.  
  11. func main() { 
  12.     go f() 
  13.     g() 

有可能 g 会接连打印2和0两个值。

双检查锁是为了降低同步造成的开销。举个例子, twoprint 方法可能会被误写成

  1. var a string 
  2. var done bool 
  3.  
  4. func setup() { 
  5.     a = "hello, world" 
  6.     done = true 
  7.  
  8. func doprint() { 
  9.     if !done { 
  10.         once.Do(setup) 
  11.     } 
  12.     print(a) 
  13.  
  14. func twoprint() { 
  15.     go doprint() 
  16.     go doprint() 

因为没有任何机制保证,协程观察到done为true的同时可以观测到a为 hello, world ,其中有一个 doprint 可能会输出空字符。

另外一个例子

  1. var a string 
  2. var done bool 
  3.  
  4. func setup() { 
  5.     a = "hello, world" 
  6.     done = true 
  7.  
  8. func main() { 
  9.     go setup() 
  10.     for !done { 
  11.     } 
  12.     print(a) 

和以前一样,不能保证在 main 中,观察对 done 的写入意味着观察对 a 的写入,因此该程序也可以打印一个空字符串。 更糟糕的情况下,由于两个线程之间没有同步事件,因此无法保证 main 会观察到对 done 的写入。 main 中的循环会一直死循环。

下面是该例子的一个更微妙的变体

  1. type T struct { 
  2.     msg string 
  3.  
  4. var g *T 
  5.  
  6. func setup() { 
  7.     t := new(T) 
  8.     t.msg = "hello, world" 
  9.     g = t 
  10.  
  11. func main() { 
  12.     go setup() 
  13.     for g == nil { 
  14.     } 
  15.     print(g.msg) 

尽管 main 观测到g不为nil,但是也没有任何机制保证可以读取到t.msg。

 

在上述例子中,解决方案都是相同的:请使用显式的同步机制。

责任编辑:张燕妮 来源: Go语言中文网
相关推荐

2020-02-28 14:48:51

结构系统程序

2021-05-06 19:20:05

Java内存模型

2022-07-10 20:49:57

javaVolatile线程

2024-11-18 16:37:35

JMMJava内存模型

2021-09-01 10:50:25

云计算云计算环境云应用

2016-11-11 00:39:59

Java可见性机制

2018-07-19 14:34:48

数据中心监控网络

2011-11-29 13:09:02

2024-02-27 17:46:25

并发程序CPU

2022-03-24 08:02:39

网络安全端点

2024-02-18 13:34:42

云计算

2023-06-13 08:29:18

网络可见性Cato

2011-07-29 11:04:52

2020-08-25 09:51:40

Android 11开发者软件

2013-08-27 09:17:15

软件定义网络SDN网络可见性

2021-12-22 11:15:04

云计算混合云公有云

2018-12-18 14:08:01

Java内存volatile

2018-05-26 16:01:37

2023-04-06 15:47:23

2016-07-04 08:19:13

混合IT网络问题SaaS
点赞
收藏

51CTO技术栈公众号