Introduction
Channels in Go are used to communicate between goroutines and synchronize their execution. Channels can be either unbuffered or buffered. Unbuffered channels block the sender and receiver until both are ready, while buffered channels allow a specified number of values to be sent and received without blocking. In this chapter, you will learn the basics of using buffered channels in Go, including creating buffered channels, sending and receiving values, and common use cases.
Creating Buffered Channels
To create a buffered channel, use the make
function with a second argument specifying the buffer size.
Example: Creating a Buffered Channel
package main
import (
"fmt"
)
func main() {
ch := make(chan int, 3) // Create a buffered channel with a capacity of 3
ch <- 1 // Send value to the channel
ch <- 2
ch <- 3
fmt.Println(<-ch) // Receive value from the channel
fmt.Println(<-ch)
fmt.Println(<-ch)
}
In this example, a buffered channel ch
with a capacity of 3 is created. Values are sent to the channel without blocking until the buffer is full.
Sending and Receiving Values
Buffered channels allow sending and receiving values without blocking, up to the buffer’s capacity.
Example: Non-Blocking Send and Receive
package main
import (
"fmt"
)
func main() {
ch := make(chan int, 2) // Create a buffered channel with a capacity of 2
ch <- 1 // Send value to the channel
ch <- 2
fmt.Println(<-ch) // Receive value from the channel
fmt.Println(<-ch)
ch <- 3
fmt.Println(<-ch)
}
In this example, values are sent and received from the buffered channel ch
without blocking, as long as the buffer is not full or empty.
Channel Blocking
A buffered channel blocks the sending operation if the buffer is full and blocks the receiving operation if the buffer is empty.
Example: Blocking Behavior
package main
import (
"fmt"
"time"
)
func main() {
ch := make(chan int, 1) // Create a buffered channel with a capacity of 1
ch <- 1 // Send value to the channel
go func() {
time.Sleep(2 * time.Second)
fmt.Println(<-ch) // Receive value from the channel after 2 seconds
}()
ch <- 2 // This send operation will block until the value is received
fmt.Println("Sent 2 to channel")
}
In this example, the second send operation to the channel ch
blocks until the value is received after 2 seconds.
Using Buffered Channels with Goroutines
Buffered channels are useful for coordinating work between goroutines without causing them to block each other unnecessarily.
Example: Worker Pool
package main
import (
"fmt"
"time"
)
func worker(id int, jobs <-chan int, results chan<- int) {
for j := range jobs {
fmt.Printf("Worker %d started job %d\n", id, j)
time.Sleep(time.Second) // Simulate work
fmt.Printf("Worker %d finished job %d\n", id, j)
results <- j * 2
}
}
func main() {
jobs := make(chan int, 5)
results := make(chan int, 5)
for w := 1; w <= 3; w++ {
go worker(w, jobs, results)
}
for j := 1; j <= 5; j++ {
jobs <- j
}
close(jobs)
for a := 1; a <= 5; a++ {
fmt.Println("Result:", <-results)
}
}
In this example, a worker pool is implemented using buffered channels. The jobs
channel holds jobs to be processed, and the results
channel holds the results. Workers process jobs concurrently without blocking each other.
Select Statement with Buffered Channels
The select
statement allows a goroutine to wait on multiple communication operations, which can be useful when working with buffered channels.
Example: Select with Buffered Channels
package main
import (
"fmt"
"time"
)
func main() {
ch1 := make(chan int, 1)
ch2 := make(chan int, 1)
ch1 <- 1
ch2 <- 2
select {
case msg1 := <-ch1:
fmt.Println("Received from ch1:", msg1)
case msg2 := <-ch2:
fmt.Println("Received from ch2:", msg2)
case <-time.After(time.Second):
fmt.Println("Timeout")
}
}
In this example, the select
statement waits for communication operations on ch1
and ch2
. The operation that can proceed first is executed.
Conclusion
Buffered channels in Go provide a powerful mechanism for communicating between goroutines without blocking, up to the buffer’s capacity. By understanding how to create and use buffered channels, you can effectively coordinate work between goroutines and improve the concurrency of your programs. The select
statement further enhances your ability to handle multiple communication operations efficiently.