Condition Variables
Nov 03, 2014
Condition variables let you block until a condition is met. For example, let's say that we're writing a little TCP server that can have up to MAX_CLIENTS connected. We might start with:
import (
"net"
"sync"
)
type Server struct {
sync.Mutex
clients int
}
func (s *Server) Listen(address string) {
l, err := net.Listen("tcp", address)
if err != nil {
panic(err)
}
defer l.Close()
for {
conn, err := l.Accept()
if err != nil {
continue
}
s.Lock()
s.clients++
s.Unlock()
go s.handleClient(conn)
}
}
func (s *Server) handleClient(conn net.Conn) {
defer s.disconnected()
for {
}
}
func (s *Server) disconnected() {
s.Lock()
s.clients--
s.Unlock()
}
One way to limit the total number of clients would be to check the value of s.clients
within a loop:
for {
s.Lock()
for s.clients == MAX_CLIENTS {
s.Unlock()
time.Sleep(time.Second)
s.Lock()
}
s.Unlock()
conn, err := l.Accept()
...
}
A more elegant solution is to use a condition variable. Condition variables provide a simple mechanism which our goroutines can use to signal a change to s.clients
. First, we define the condition variable:
import (
"net"
"sync"
"sync/atomic"
)
type Server struct {
clients uint64
cond *sync.Cond
}
Condition variable are made up of their own mutex. To iniate one, we'd do:
s := &Server{
cond: &sync.Cond{L: &sync.Mutex{}},
}
Next, instead of the above for
spin, we can Wait
for a signal:
for {
s.cond.L.Lock()
for s.listeners == MAX_CLIENTS {
s.cond.Wait()
}
s.cond.L.Unlock()
conn, err := l.Accept()
...
}
And we change our disconnected
method:
func (s *Server) disconnected() {
s.cond.L.Lock()
s.clients--
s.cond.L.Unlock()
s.cond.Signal()
}
There are a couple interesting things in the above code. First of all, notice the locking and unlocking around the call to Wait
. It might seem like we're locking for a very long time. But Wait
unlocks L
on entry and relocks L
on exit. This results in much cleaner code -- you lock and unlock normally, without being locked while you wait.
Also, notice that we're still checking our condition inside of a loop. This is because the state of s.clients
could be changed by a different goroutine between the time that the signal is sent and our code exiting Wait
. (In this specific example, when the blocked goroutine is also the only one that can increment s.clients
, the for loop is unecessary. But I wanted to show the for loop example anyways because it's more complete and more common).