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 小时前
DJango项目
后端·python·django
Mr Aokey4 小时前
Spring MVC参数绑定终极手册:单&多参/对象/集合/JSON/文件上传精讲
java·后端·spring
地藏Kelvin4 小时前
Spring Ai 从Demo到搭建套壳项目(二)实现deepseek+MCP client让高德生成昆明游玩4天攻略
人工智能·spring boot·后端
菠萝015 小时前
共识算法Raft系列(1)——什么是Raft?
c++·后端·算法·区块链·共识算法
长勺5 小时前
Spring中@Primary注解的作用与使用
java·后端·spring
小奏技术6 小时前
基于 Spring AI 和 MCP:用自然语言查询 RocketMQ 消息
后端·aigc·mcp
编程轨迹6 小时前
面试官:如何在 Java 中读取和解析 JSON 文件
后端
lanfufu6 小时前
记一次诡异的线上异常赋值排查:代码没错,结果不对
java·jvm·后端
编程轨迹6 小时前
如何在 Java 中实现 PDF 与 TIFF 格式互转
后端
编程轨迹6 小时前
面试官:你知道如何在 Java 中创建对话框吗
后端