boxmoe_header_banner_img

Hello! 欢迎来到悠悠畅享网!

文章导读

Golang的channel如何使用 讲解无缓冲与有缓冲通道的语法差异


avatar
站长 2025年8月6日 9

无缓冲通道需同步收发,有缓冲通道允许暂存数据。无缓冲通道声明为ch := make(chan int),发送和接收操作必须同时发生,适合同步控制;有缓冲通道声明为ch := make(chan int, 3),发送可在缓冲未满时进行,接收可在缓冲非空时进行,适合数据暂存和流量控制;常见错误包括顺序不当导致死锁、忘记启动另一方处理;建议优先使用无缓冲通道,合理设置缓冲大小,注意关闭通道的方式。

Golang的channel如何使用 讲解无缓冲与有缓冲通道的语法差异

在 Go 语言中,channel 是实现 goroutine 之间通信的重要工具。理解无缓冲和有缓冲通道的使用方式及其语法差异,是掌握并发编程的关键一步。

Golang的channel如何使用 讲解无缓冲与有缓冲通道的语法差异


无缓冲通道:发送和接收必须同时发生

无缓冲通道(unbuffered channel)没有存储空间,发送方必须等到有接收方准备好才能完成发送操作,反之亦然。这种“同步”特性让它非常适合用于协程之间的同步控制。

声明方式如下:

立即学习go语言免费学习笔记(深入)”;

Golang的channel如何使用 讲解无缓冲与有缓冲通道的语法差异

ch := make(chan int)

举个简单例子:

ch := make(chan int)  go func() {     fmt.Println("sending value")     ch <- 42 // 发送数据到通道 }()  fmt.Println("waiting to receive") fmt.Println(<-ch) // 接收数据

这里,子 goroutine 向通道发送值时会阻塞,直到主 goroutine 执行到

<-ch

才能继续。如果顺序反过来,程序就会死锁。

Golang的channel如何使用 讲解无缓冲与有缓冲通道的语法差异

常见错误

  • 在一个 goroutine 中先发送再接收,但没有其他 goroutine 处理,会导致死锁。
  • 忘记启动接收方或发送方,导致程序卡住。

有缓冲通道:可以暂存数据

有缓冲通道(buffered channel)允许一定数量的数据在没有接收方的情况下被缓存。只有当缓冲区满了之后,发送才会阻塞。

声明方式是在

make

函数中指定容量:

ch := make(chan int, 3) // 容量为3的缓冲通道

比如:

ch := make(chan int, 2)  ch <- 1 ch <- 2>

在这个例子中,前两次发送不会阻塞,因为缓冲区还有空间。第三次发送如果没有接收动作,就会造成阻塞。

典型用途

  • 控制并发数量(如信号量模式)
  • 暂存一批数据后再处理
  • 避免频繁创建 goroutine 的开销

使用上的关键区别总结

特性 无缓冲通道 有缓冲通道
声明方式 @@######@@ @@######@@
是否需要同步收发 否(只要缓冲未满/非空)
主要用途 协程同步、精确控制流程 数据暂存、流量控制
死锁风险 更高 相对较低

实际开发中的一些小建议

  • 优先考虑无缓冲通道:除非你明确需要缓存功能,否则无缓冲通道更容易帮助你发现并发逻辑中的问题。
  • 缓冲大小别设太大:设置过大容易掩盖性能瓶颈,也容易让程序行为变得难以预测。
  • 注意关闭通道的方式:关闭通道只能由发送方来做,重复关闭会导致 panic。
  • 用 range 遍历通道时小心:只有在通道关闭后循环才会退出,记得确保关闭操作一定会执行。

基本上就这些。两种通道的核心区别在于是否具备缓存能力,语法上也只是 make 时有没有指定容量的区别。但在实际使用中,它们的行为差异很大,选错可能引发死锁或者资源浪费。

make(chan T)
make(chan T, N)



评论(已关闭)

评论已关闭