這是一個建立於 的文章,其中的資訊可能已經有所發展或是發生改變。
http://golang.org/ref/mem
The Go Memory Model
Version of March 6, 2012
-
Introduction
-
Happens Before
-
Synchronization
-
Initialization
-
Goroutine creation
-
Goroutine destruction
-
Channel communication
-
Locks
-
Once
-
Incorrect synchronization
|
|
Introduction
The Go memory model specifies the conditions under whichreads of a variable in one goroutine can be guaranteed toobserve values produced by writes to the same variable in a different goroutine.
Happens Before
Within a single goroutine, reads and writes must behaveas if they executed in the order specified by the program.That is, compilers and processors may reorder the reads and writesexecuted within a single goroutine only when the reorderingdoes not change the behavior within that goroutineas defined by the language specification.Because of this reordering, the execution order observedby one goroutine may differ from the order perceivedby another. For example, if one goroutineexecutesa = 1; b = 2;
, another might observethe updated value of b
before the updated value ofa
.
To specify the requirements of reads and writes, we definehappens before, a partial order on the executionof memory operations in a Go program. If evente1 happensbefore event e2, then we say that e2 happens aftere1.Also, if e1 does not happen beforee2 and does not happenafter e2, then we say that e1 ande2 happen concurrently.
Within a single goroutine, the happens-before order is theorder expressed by the program.
A read r of a variable v
is allowed to observe a writew to v
if both of the following hold:
- r does not happen before w.
- There is no other write w' to
v
that happens afterw but before r.
To guarantee that a read r of a variable v
observes aparticular writew to v
, ensure that w is the onlywriter is allowed to observe.That is, r isguaranteed to observe w if both of the following hold:
- w happens before r.
- Any other write to the shared variable
v
either happens before w or after r.
This pair of conditions is stronger than the first pair;it requires that there are no other writes happeningconcurrently withw or r.
Within a single goroutine,there is no concurrency, so the two definitions are equivalent:a readr observes the value written by the most recent write w to v
.When multiple goroutines access a shared variable v
,they must use synchronization events to establishhappens-before conditions that ensure reads observe thedesired writes.
The initialization of variable v
with the zero valuefor v
's type behaves as a write in the memory model.
Reads and writes of values larger than a single machine wordbehave as multiple machine-word-sized operations in anunspecified order.
Synchronization
Initialization
Program initialization runs in a single goroutine,but that goroutine may create other goroutines,which run concurrently.
If a package p
imports package q
, the completion ofq
'sinit
functions happens before the start of any of p
's.
The start of the function main.main
happens afterallinit
functions have finished.
Goroutine creation
The go
statement that starts a new goroutinehappens before the goroutine's execution begins.
For example, in this program:
var a stringfunc f() {print(a)}func hello() {a = "hello, world"go f()}
calling hello
will print "hello, world"
at some point in the future (perhaps afterhello
has returned).
Goroutine destruction
The exit of a goroutine is not guaranteed to happen beforeany event in the program. For example, in this program:
var a stringfunc hello() {go func() { a = "hello" }()print(a)}
the assignment to a
is not followed byany synchronization event, so it is not guaranteed to beobserved by any other goroutine.In fact, an aggressive compiler might delete the entirego
statement.
If the effects of a goroutine must be observed by another goroutine,use a synchronization mechanism such as a lock or channelcommunication to establish a relative ordering.
Channel communication
Channel communication is the main method of synchronizationbetween goroutines. Each send on a particular channelis matched to a corresponding receive from that channel,usually in a different goroutine.
A send on a channel happens before the correspondingreceive from that channel completes.
This program:
var c = make(chan int, 10)var a stringfunc f() {a = "hello, world"c <- 0}func main() {go f()<-cprint(a)}
is guaranteed to print "hello, world"
. The write to a
happens before the send onc
, which happens beforethe corresponding receive on c
completes, which happens beforetheprint
.
The closing of a channel happens before a receive that returns a zero valuebecause the channel is closed.
In the previous example, replacingc <- 0
with close(c)
yields a program with the same guaranteed behavior.
A receive from an unbuffered channel happens beforethe send on that channel completes.
This program (as above, but with the send and receive statements swapped andusing an unbuffered channel):
var c = make(chan int)var a stringfunc f() {a = "hello, world"<-c}
func main() {go f()c <- 0print(a)}
is also guaranteed to print "hello, world"
. The write to a
happens before the receive onc
, which happens beforethe corresponding send on c
completes, which happensbefore theprint
.
If the channel were buffered (e.g., c = make(chan int, 1)
)then the program would not be guaranteed to print"hello, world"
. (It might print the empty string,crash, or do something else.)
Locks
The sync
package implements two lock data types,sync.Mutex
andsync.RWMutex
.
For any sync.Mutex
or sync.RWMutex
variablel
and n < m,call n of l.Unlock()
happens before callm of l.Lock()
returns.
This program:
var l sync.Mutexvar a stringfunc f() {a = "hello, world"l.Unlock()}func main() {l.Lock()go f()l.Lock()print(a)}
is guaranteed to print "hello, world"
.The first call to l.Unlock()
(inf
) happensbefore the second call to l.Lock()
(in main
) returns,which happens before the print
.
For any call to l.RLock
on a sync.RWMutex
variablel
,there is an n such that the l.RLock
happens (returns) after calln tol.Unlock
and the matching l.RUnlock
happensbefore calln+1 to l.Lock
.
Once
The sync
package provides a safe mechanism forinitialization in the presence of multiple goroutinesthrough the use of theOnce
type.Multiple threads can execute once.Do(f)
for a particularf
,but only one will run f()
, and the other calls blockuntilf()
has returned.
A single call of f()
from once.Do(f)
happens (returns) before any call ofonce.Do(f)
returns.
In this program:
var a stringvar once sync.Oncefunc setup() {a = "hello, world"}func doprint() {once.Do(setup)print(a)}func twoprint() {go doprint()go doprint()}
calling twoprint
causes "hello, world"
to be printed twice.The first call totwoprint
runs setup
once.
Incorrect synchronization
Note that a read r may observe the value written by a writewthat happens concurrently with r.Even if this occurs, it does not imply that reads happening afterrwill observe writes that happened before w.
In this program:
var a, b intfunc f() {a = 1b = 2}func g() {print(b)print(a)}func main() {go f()g()}
it can happen that g
prints 2
and then 0
.
This fact invalidates a few common idioms.
Double-checked locking is an attempt to avoid the overhead of synchronization.For example, thetwoprint
program might beincorrectly written as:
var a stringvar done boolfunc setup() {a = "hello, world"done = true}func doprint() {if !done {once.Do(setup)}print(a)}func twoprint() {go doprint()go doprint()}
but there is no guarantee that, in doprint
, observing the write todone
implies observing the write to a
. Thisversion can (incorrectly) print an empty stringinstead of"hello, world"
.
Another incorrect idiom is busy waiting for a value, as in:
var a stringvar done boolfunc setup() {a = "hello, world"done = true}func main() {go setup()for !done {}print(a)}
As before, there is no guarantee that, in main
,observing the write todone
implies observing the write to a
, so this program couldprint an empty string too.Worse, there is no guarantee that the write todone
will everbe observed by main
, since there are no synchronizationevents between the two threads. The loop inmain
is notguaranteed to finish.
There are subtler variants on this theme, such as this program.
type T struct {msg string}var g *Tfunc setup() {t := new(T)t.msg = "hello, world"g = t}func main() {go setup()for g == nil {}print(g.msg)}
Even if main
observes g != nil
and exits its loop,there is no guarantee that it will observe the initializedvalue forg.msg
.
In all these examples, the solution is the same:use explicit synchronization.