Go Concurrency: Mutexes vs Channels with Examples
Can we use Buffered and Unbuffered Channels instead of Mutexes to handle synchronization in Go? Let’s know how.
Introduction
When building concurrent applications in Go, synchronization is crucial to ensure that shared data is accessed safely. In Go, Mutexes and Channels are the primary tools used for synchronization.
Motivation
I am learning Golang these days, and I came across an interesting problem in which I need to build a counter that is safe to use concurrently.
However, in the mentioned article, the author solved the problem using one approach: Mutexes. But I was wondering if I could solve the same problem using the Buffered Channels and the Unbuffered Channels.
Have a look at the counter code:
package main
type Counter struct {
count int
}
func (c *Counter) Inc() {
c.count++
}
func (c *Counter) Value() int {
return c.count
}
Please find the code here.
To ensure our code is safe to use concurrently, let’s dive into writing some tests.
Let’s start with the simplest approach first.
1) Mutexes
A Mutex (short for “mutual exclusion”) is a synchronization primitive that ensures only one goroutine can access a critical section of code at a time.
It provides a lock mechanism, when one goroutine locks a mutex, other goroutines trying to lock it will block until the mutex is unlocked. So it is typically used when you need to protect shared variables or resources from race conditions.
package main
import (
"sync"
"testing"
)
func TestCounter(t *testing.T) {
t.Run("using mutexes and wait groups", func(t *testing.T) {
counter := Counter{}
wantedCount := 1000
var wg sync.WaitGroup
var mut sync.Mutex
wg.Add(wantedCount)
for i := 0; i < wantedCount; i++ {
go func() {
mut.Lock()
counter.Inc()
mut.Unlock()
wg.Done()
}()
}
wg.Wait()
if counter.Value() != wantedCount {
t.Errorf("got %d, want %d", counter.Value(), wantedCount)
}
})
}
sync.WaitGroup
Wait Groups are used to track the completion of all goroutines.sync.Mutex
is used to prevent concurrent access to the sharedcounter
by multiple goroutines at the same time (to avoid a race condition).The loop starts
1000
goroutines. Each goroutine does the following:mut.Lock()
: First locks the mutex before accessing thecounter
and calling itsInc()
method. This ensures that only one goroutine can increment the counter at a time, preventing race conditions.counter.Inc
()
: Only one goroutine can call this method at a time, thanks to the mutex lock.mut.Unlock()
: Unlocks the mutex after the counter is incremented. This allows other goroutines to acquire the lock and perform their own increment operations.wg.Done()
: Callswg.Done()
to signal that it has completed its work (incrementing the counter). This reduces theWaitGroup
counter by one.
wg.Wait()
: This makes the main goroutine wait until all1000
worker goroutines have been completed. TheWait()
method blocks until theWaitGroup
counter reaches zero (when allwg.Done()
calls have been made).
2) Buffered Channels
Channels are Go’s way of allowing goroutines to communicate with each other safely. They enable the transfer of data between goroutines and provide synchronization by controlling access to the data being passed.
Having that said, in our example, we will leverage this fact in channels to block goroutines and let just one goroutine access the shared data.
In this case, Buffered Channels have a fixed capacity, meaning they can hold a predefined number of elements before blocking the sender. The sender only blocks when the buffer is full.
package main
import (
"sync"
"testing"
)
func TestCounter(t *testing.T) {
t.Run("using buffered channels and wait groups", func(t *testing.T) {
counter := Counter{}
wantedCount := 1000
var wg sync.WaitGroup
wg.Add(wantedCount)
ch := make(chan struct{}, 1)
ch <- struct{}{}
for i := 0; i < wantedCount; i++ {
go func() {
<-ch
counter.Inc()
ch <- struct{}{}
wg.Done()
}()
}
wg.Wait()
if counter.Value() != wantedCount {
t.Errorf("got %d, want %d", counter.Value(), wantedCount)
}
})
}
ch := make(chan struct{}, 1)
: A buffered channelch
with a capacity of1
is created. The buffer size of1
allows just one goroutine to write to the channel at a time.chan struct{}
: Used an emptystruct
instead of other types (likeint
,bool
, etc.), because it takes no memory. It has a size of 0 bytes. This makes it ideal for scenarios like signaling where you don’t need to pass any actual data, just a signal. On the other hand, other types (likeint
,bool
, etc.) would consume more memory, which isn’t necessary when you just need a signal.ch <- struct{}{}
: The first signal is sent to the buffered channel from the main function to allow the first goroutine to start. Since the channel has a capacity of1
, this operation doesn’t block and enables the first worker goroutine to proceed.The loop starts
1000
goroutines. Each goroutine does the following:<-ch
Waits for a signal coming from the previously finished goroutine or the first signal in case of the first loop (the previous point) to increment the counter.counter.Inc
()
: Once the signal is received, the counter is incremented by1
.ch <- struct{}{}
: After incrementing, the goroutine sends a signal to the channel, allowing the next goroutine to proceed.
3) Unbuffered Channels
These channels do not have a buffer. They block the sender until the receiver is ready to receive the data. This provides strict synchronization where data is passed between goroutines one at a time.
package main
import (
"sync"
"testing"
)
func TestCounter(t *testing.T) {
t.Run("using unbuffered channels and wait groups", func(t *testing.T) {
counter := Counter{}
wantedCount := 1000
var wg sync.WaitGroup
wg.Add(wantedCount)
ch := make(chan struct{})
go func() {
ch <- struct{}{}
}()
for i := 0; i < wantedCount; i++ {
go func() {
<-ch
counter.Inc()
go func() {
ch <- struct{}{}
}()
wg.Done()
}()
}
wg.Wait()
if counter.Value() != wantedCount {
t.Errorf("got %d, want %d", counter.Value(), wantedCount)
}
})
}
ch := make(chan struct{})
: An Unbuffered Channel is created with the typestruct{}
. The typestruct{}
is used because it doesn’t hold any data, and the channel is used purely for signaling.go func() { ch <- struct{}{} } ()
: This anonymous goroutine sends an initial signal to the channel, which allows the first worker goroutine to start executing. The channel will be empty initially, so the first signal unlocks the first goroutine.The loop starts
1000
goroutines. Each goroutine does the following:<-ch
: Waits for the signal (thestruct{}{}
value) from the channel. Since the channel is unbuffered, the goroutine is blocked until another goroutine sends a signal to the channel. This ensures that only one goroutine will run at a time.counter.Inc
()
: Increments the counter by1
once the signal is received. This is the critical section where the counter is updated, and it is protected by the signaling mechanism, so only one goroutine can increment the counter at any time.go func() { ch <- struct{}{} } ()
: Sends a signal back to the channel after incrementing the counter, allowing the next waiting goroutine to start. The sending of the signal is done in a separate goroutine to ensure that the channel operation (ch <- struct{}{}
) doesn’t block the outer goroutine. This ensures that the channel operations do not cause a deadlock.
4) Buffered Channels without Wait Groups
After solving this problem using the aforementioned solutions, I asked myself, “Can I solve it without Wait Groups?”. Actually, I came up with two solutions.
In fact, Wait Groups make the main function wait until all the sub-goroutines are completed. So I thought that we could use either an infinite loop that breaks in a condition or we can use another channel to track the goroutines’ completion.
Let’s jump into the code using the infinite loop.
package main
import (
"sync"
"testing"
)
func TestCounter(t *testing.T) {
t.Run("using buffered channels without wait groups (infinite loop)", func(t *testing.T) {
counter := Counter{}
wantedCount := 1000
ch := make(chan struct{}, 1)
ch <- struct{}{}
for i := 0; i < wantedCount; i++ {
go func() {
<-ch
counter.Inc()
ch <- struct{}{}
}()
}
for {
if counter.Value() == wantedCount {
break
}
}
if counter.Value() != wantedCount {
t.Errorf("got %d, want %d", counter.Value(), wantedCount)
}
})
}
As you see, I am using another waiting channel
wc
that will close “signal” by the end of the last goroutineclose(wc)
.<-wc
During the work of the goroutines, this receiver blocks the code until receiving a signal from its senderwc
.close(wc)
By closing thewc
channel, it sends a signal to the receiver<-wc
that it is done.At this time, the
wc
channel releases the block which makes us guarantee that all goroutines are completed.
Conclusion
In this article, we explored different ways to solve the problem of building a counter that is safe to use concurrently in Go. While the article we referenced implemented the solution using Mutexes, we also discussed alternative approaches using Buffered and Unbuffered Channels.
Understanding these tools and when to use them is key to writing efficient and safe concurrent Go programs.
So, whether you choose Mutexes, Buffered, or Unbuffered Channels, mastering synchronization in Go is essential, it will help you build robust applications that can handle concurrency with ease.
Resources
In fact, this article is inspired by the Sync Chapter in Learn Go with tests.
Think about it
If you liked this article please rate and share it to spread the word, really, that encourages me a lot to create more content like this.
You can check out more articles as well:
Thanks a lot for staying with me up till this point. I hope you enjoy reading this article.