Go 并发编程基础:通道(Channel)的使用

在 Go 中,Channel 是 Goroutine 之间通信的核心机制。它提供了一个线程安全的通信方式,用于在多个 Goroutine 之间传递数据,从而实现高效的并发编程。

本章将介绍 Channel 的基本概念、用法、缓冲、关闭机制以及 select 的使用。


一、Channel 简介

Channel 是 Go 语言中实现 "通信顺序进程"(CSP)模型的关键组件。

  • • 用于在多个 Goroutine 之间传递数据
  • • 本质上是一个 先进先出 的队列
  • • 保证并发安全

通道类型的声明格式:

go 复制代码
var ch chan 数据类型

二、Channel 的创建与基本使用

go 复制代码
ch := make(chan int) // 创建一个传输 int 类型的无缓冲通道

发送与接收数据:

go 复制代码
ch <- 10     // 发送数据
x := <- ch   // 接收数据

注意:无缓冲通道发送和接收都是阻塞的,直到对方准备好。

示例:

go 复制代码
func worker(ch chan string) {
    msg := <-ch
    fmt.Println("接收到:", msg)
}

func main() {
    ch := make(chan string)
    go worker(ch)
    ch <- "Hello, Channel"
}

三、带缓冲的 Channel

go 复制代码
ch := make(chan int, 3) // 创建一个缓冲区大小为3的通道

特点:

  • • 发送操作在缓冲区满时阻塞
  • • 接收操作在缓冲区空时阻塞
r 复制代码
ch <- 1
ch <- 2
fmt.Println(<-ch) // 输出 1

四、通道的关闭

scss 复制代码
close(ch)
  • • 关闭通道后不能再发送数据,但仍可接收剩余数据
  • • 读取已关闭通道不会阻塞,返回类型零值
  • • 可以用 v, ok := <-ch 判断通道是否关闭

示例:

go 复制代码
ch := make(chan int, 2)
ch <- 10
ch <- 20
close(ch)

for v := range ch {
    fmt.Println(v)
}

五、单向通道

可以将通道限制为只发送只接收

go 复制代码
func send(ch chan<- int) {
    ch <- 100
}

func recv(ch <-chan int) {
    fmt.Println(<-ch)
}

六、使用 select 同时监听多个 Channel

go 复制代码
select {
case msg1 := <-ch1:
    fmt.Println("ch1:", msg1)
case msg2 := <-ch2:
    fmt.Println("ch2:", msg2)
default:
    fmt.Println("无数据可读")
}

特点:

  • select 会等待多个通道中的一个准备好
  • • 如果多个都准备好了,则随机选择一个执行
  • • 使用 default 实现非阻塞操作

七、超时控制

结合 time.After() 实现超时机制:

css 复制代码
select {
case data := <-ch:
    fmt.Println("收到数据:", data)
case <-time.After(2 * time.Second):
    fmt.Println("超时未收到数据")
}

八、通道的常见应用场景

场景 描述
Goroutine 通信 不共享内存,通过通道交换数据
控制并发数量(限流) 使用带缓冲通道实现并发任务限制
Worker Pool(协程池) 多个 worker 从任务通道中取任务处理
信号通知/任务完成通知 通道发送空 struct{} 信号实现同步或退出控制
超时控制 / 取消上下文 select + time.After / context 控制并发

九、小结

  • • Channel 是 Go 并发通信的核心工具
  • • 区分无缓冲/有缓冲通道,理解其阻塞机制
  • • 使用 close 安全关闭通道
  • select 可实现多路复用、超时控制等高级场景

相关推荐
桦说编程1 天前
简单方法实现子任务耗时统计
java·后端·监控
盖世英雄酱581361 天前
物品超领取损失1万事故复盘(一)
java·后端
凌览1 天前
别再死磕 Nginx!http-proxy-middleware 低配置起飞
前端·后端
拾玖不会code1 天前
简单分表场景下的业务发散思考:分表如何保证丝滑?
后端
CryptoRzz1 天前
印度尼西亚(IDX)股票数据对接开发
java·后端·websocket·web3·区块链
咕白m6251 天前
通过 C# 快速生成二维码 (QR code)
后端·.net
踏浪无痕1 天前
架构师如何学习 AI:三个月掌握核心能力的务实路径
人工智能·后端·程序员
小毅&Nora1 天前
【后端】【SpringBoot】① 源码解析:从启动到优雅关闭
spring boot·后端·优雅关闭
嘻哈baby1 天前
从TIME_WAIT爆炸到端口耗尽:Linux短连接服务排查与优化
后端
开心就好20251 天前
iOS应用性能监控全面解析:CPU、内存、FPS、卡顿与内存泄漏检测
后端