GoLang channel管道

🎯 一、channel 学习全景图

你可以把 channel 学成这 5 层:

复制代码
基础语法 → 使用模式 → 并发控制 → 底层原理 → 实战设计

🧠 二、第一层:基础语法(必须熟练)

1️⃣ 创建

复制代码
ch := make(chan int)        // 无缓冲
ch := make(chan int, 10)    // 有缓冲

2️⃣ 发送 / 接收

复制代码
ch <- 1      // 发送
x := <-ch    // 接收

3️⃣ 关闭

复制代码
close(ch)

4️⃣ 判断关闭

复制代码
v, ok := <-ch

5️⃣ range

复制代码
for v := range ch {
    fmt.Println(v)
}

👉 重点:必须 close,否则死循环


⚙️ 三、第二层:核心特性(必须理解)


1️⃣ 阻塞机制(最重要)

👉 channel 的本质是:同步机制

无缓冲 channel

复制代码
ch := make(chan int)

go func() {
    ch <- 1
}()

<-ch

👉 发送必须等接收


有缓冲 channel

复制代码
ch := make(chan int, 2)

👉 不满不阻塞


2️⃣ FIFO(先进先出)

👉 channel 内部是队列


3️⃣ 单向 channel(面试加分)

复制代码
func send(ch chan<- int)
func recv(ch <-chan int)

👉 限制方向,提高安全性


🔄 四、第三层:select 多路复用(重点)

复制代码
select {
case v := <-ch1:
    fmt.Println(v)
case ch2 <- 10:
    fmt.Println("send")
default:
    fmt.Println("no ready")
}

🔥 关键点

  • 随机选择一个可执行 case

  • 没有 ready:

    • 有 default → 执行 default

    • 没 default → 阻塞


⏱ 超时控制

复制代码
select {
case <-ch:
case <-time.After(time.Second):
    fmt.Println("timeout")
}

🚀 五、第四层:常见使用模式(非常重要)


1️⃣ 生产者-消费者

复制代码
ch := make(chan int, 10)

go func() {
    for i := 0; i < 10; i++ {
        ch <- i
    }
    close(ch)
}()

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

2️⃣ worker pool(面试高频)

复制代码
jobs := make(chan int, 100)

for i := 0; i < 5; i++ {
    go func() {
        for job := range jobs {
            fmt.Println(job)
        }
    }()
}

3️⃣ fan-in(合并多个 channel)

复制代码
func fanIn(ch1, ch2 <-chan int) <-chan int

4️⃣ fan-out(广播)

👉 一个 channel → 多个 worker


5️⃣ 通知模式

复制代码
done := make(chan struct{})

👉 用于结束 goroutine


🧠 六、第五层:底层原理(面试高频)


📦 channel 本质结构

复制代码
type hchan struct {
    buf      // 环形队列
    sendq    // 发送等待队列
    recvq    // 接收等待队列
    lock     // 互斥锁
}

📤 发送流程

  1. 有接收者 → 直接交付

  2. 有缓冲 → 放入 buffer

  3. 满了 → 阻塞(进入 sendq)


📥 接收流程

  1. 有数据 → 直接取

  2. 有发送者 → 直接拿

  3. 没数据 → 阻塞(进入 recvq)


🔐 为什么线程安全?

👉 因为:

  • 内部加锁

  • 调度器控制 goroutine


⚠️ 七、常见问题(必须掌握)


❌ 死锁

复制代码
ch := make(chan int)
ch <- 1 // 没人接收

❌ goroutine 泄漏

复制代码
go func() {
    <-ch // 永远阻塞
}()

❌ close 使用错误

复制代码
close(ch)
close(ch) // panic

❌ 向已关闭 channel 发送

👉 直接 panic


⚔️ 八、channel vs mutex(面试必问)

对比 channel mutex
用途 通信 共享数据
安全性 需手动控制
性能 稍低 更快

👉 结论:

简单共享用锁,复杂协作用 channel


🧩 九、学习路径(你可以照着练)

第1步(1天)

  • 基本语法

  • 阻塞理解


第2步(2天)

  • select

  • timeout


第3步(3天)

  • worker pool

  • producer-consumer


第4步(进阶)

  • goroutine 泄漏

  • context + channel


🔥 十、终极总结(面试王炸)

👉 你可以这样说:

channel 是 Go 提供的 goroutine 通信机制,支持同步和异步传输,通过阻塞保证并发安全,底层由环形队列和等待队列实现,常用于任务调度、并发控制和数据流处理。

相关推荐
阿正的梦工坊1 天前
JavaScript 微任务与宏任务完全指南
开发语言·javascript·ecmascript
GetcharZp1 天前
拒绝低效!这款神器,让你的终端效率起飞 | 深度解析 fzf 终极指南
后端
知行合一。。。1 天前
Python--05--面向对象(属性,方法)
android·开发语言·python
青梅橘子皮1 天前
C语言---指针的应用以及一些面试题
c语言·开发语言·算法
自珍JAVA1 天前
高效处理Long列表与集合运算:基于RoaringBitmap的工具类解析与应用场景
后端
小码哥_常1 天前
Spring Boot项目上线秘籍:日志、监控、异常处理全攻略
后端
浅时光_c1 天前
3 shell脚本编程
linux·开发语言·bash
Evand J1 天前
【三维轨迹目标定位,CKF+RTS,MATLAB程序】基于CKF与RTS平滑的三维非线性目标跟踪(距离+方位角+俯仰角)
开发语言·matlab·目标跟踪
GreenTea1 天前
AI 时代,工程师的不可替代性在哪里
前端·人工智能·后端
朦胧之1 天前
AI 编程开发思维
前端·后端·ai编程