golang 的 channel 是并发编程的核心,用于安全高效地在 goroutine 之间通信。1. 创建 channel 使用 make 函数并指定数据类型和可选缓冲大小;2. 使用

Golang 的 channel 是并发编程中非常核心的一个概念,它让 goroutine 之间的通信变得简单、安全又高效。理解并掌握 channel 的使用,是写出稳定、高性能 Go 程序的关键之一。

创建和基本操作
Channel 在 Go 中通过
make函数创建,声明时需要指定传输的数据类型。例如:
ch := make(chan int)
这行代码创建了一个可以传递整型数据的无缓冲通道。你可以通过
<-操作符来发送或接收数据:
立即学习“go语言免费学习笔记(深入)”;

- 发送:
ch <- 10
- 接收:
num := <-ch
注意,无缓冲通道在发送和接收双方都准备好之前会阻塞。也就是说,如果只有发送方没有接收方,程序会卡住;反之亦然。
如果你希望通道能缓存一定数量的数据,可以指定容量:

ch := make(chan int, 3)
这样最多可以缓存 3 个元素,发送方不会立即阻塞,直到通道满了为止。
使用 select 实现多路复用
在实际开发中,经常需要同时处理多个通道的操作。这时候就要用到
select语句了。它的作用类似于 switch,但专门用于监听多个 channel 的读写事件。
举个例子:
select {
case msg1 := <-ch1:
fmt.Println("Received from ch1:", msg1)
case msg2 := <-ch2:
fmt.Println("Received from ch2:", msg2)
default:
fmt.Println("No message received")
}上面这个 select 会尝试从
ch1或
ch2中读取数据。哪个 channel 有数据就处理哪个;如果没有数据且有 default 分支,就执行 default。
几点需要注意:
- 如果多个 case 都准备好了,会随机选一个执行。
- 没有 default 的情况下,select 会一直等待,直到有 case 可以执行。
- 常用于超时控制、轮询多个 channel 等场景。
关闭 channel 和 range 遍历
当不再发送数据时,可以通过
close(ch)来关闭通道。关闭后不能再向通道发送数据,但仍然可以从通道接收数据,直到所有数据都被取完。
结合
range可以很方便地遍历一个被关闭的 channel:
go func() {
for i := 0; i < 5; i++ {
ch <- i
}
close(ch)
}()
for num := range ch {
fmt.Println(num)
}这里要注意:
- 不要重复关闭已经关闭的 channel,会导致 panic。
- 接收方可以通过第二个返回值判断通道是否已关闭:
num, ok := <-ch
if !ok {
fmt.Println("Channel closed")
}实际使用中的几个小技巧
- 避免死锁:确保发送和接收成对出现,尤其是在使用无缓冲 channel 时。
- 使用带缓冲的 channel 提升性能:适用于生产者消费者模型,减少不必要的阻塞。
- 配合 context 控制生命周期:比如在并发任务中,通过 context.Done() 通知各个 goroutine 停止工作。
- 合理使用 default 分支:可以在 select 中做非阻塞检查,适合心跳检测、状态上报等场景。
基本上就这些。channel 和 select 搭配起来功能强大,但用不好也容易出错。关键是理解它们的行为机制,并根据实际需求选择合适的模式。










