还记得前面的银行账户存款的例子吗?为了解决竞争问题,我们曾经使用了 sync.Mutex 来保护共享变量。
特别的,在读取 balance 变量的时候,我们是这样写的:
当时我们说到,这里加锁需要进一步讨论。这种情况也是并发编程里让人感到相当困惑的地方。
1. 例子
我们来看一个简单的例子。
仅通过分析,我们能预见的可能的结果有:
然而,实际在运行的时候,可能还会出现下面的结果:
这已经不是传统意义上的编译器和 cpu 的工作方式了。这个问题不只会出现在 Golang 里,同样也会出现在你所编写的 C/C++ 的程序里。(并发编程是一件挑战性极高的事,不是说加个锁你的程序就没问题了。)
上面的结果要怎么解释呢?
1.1 指令重排
一般来说,在一个 goroutine 里,每条语句的执行顺序都是保证的。但是,在不使用 channel 且不使用 mutex 这样的显式同步操作的时候,我们就没办法保证每条语句的执行顺序(指令重排)。
这样一来,编译器就有可能在编译这两个 goroutine 的时候,就有可能将赋值操作放到 print 语句后面。
1.2 内存同步
除了编译器指令重排可能导致这种问题外,在多核处理器中,还会有主存同步的问题。
现代处理器为了提高运行效率,每个核心都有自己的 Local Cache,并且在必要的时候,将其 commit 到内存(主存同步)。
在 Golang 里像 channel 通信,互斥操作这样的原语都会触发“主存同步”。只有这样,在一个 goroutine 里的执行结果,才能被运行在其它 cpu 或其它核心的 goroutine 所看到。
在上面的盒子里,goroutine A 和 B 都没有使用 channel 或是 mutex 这样的操作,因此有可能不会触发主存同步。如果这两个 gorotuine 运行在不同的 cpu 核心上,这样一来,两个 goroutine 所看到的内存也可能不一样(内存可见性)。
假设 goroutine B 先执行,y = 1, 并打印出 x = 0,但是 goroutine A 所看到的 y 未必就是 1. 因为 goroutine A 运行在自己的核心上,它所读取的是自己核心中的缓存,所以 goroutine 可能看到的 y 还是 0。
2. 总结
- 主存同步
- 内存一致性
- 指令重排