
在 go 中,可以通过 chan *chan t 类型实现“通道中传递通道指针”,但更推荐直接使用 chan chan t——因为通道本身是引用类型,无需额外取地址,既安全又简洁。
Go 允许将通道作为值类型参与通道通信,前提是明确指定其元素类型。例如,若需在一个通道中传输另一个通道(如用于动态分发或运行时创建子通道),应定义为 chan chan T,而非 chan *chan T。
✅ 正确写法:直接传递通道值(推荐)
func worker(in <-chan chan int) {
for ch := range in {
ch <- 42 // 向接收到的通道发送数据
}
}
func main() {
// 创建一个用于传输通道的通道
chOfChans := make(chan chan int, 1)
// 启动 worker
go worker(chOfChans)
// 创建子通道并发送过去
subChan := make(chan int, 1)
chOfChans <- subChan
// 接收结果
fmt.Println(<-subChan) // 输出: 42
close(chOfChans)
}⚠️ 不推荐:传递通道指针(chan *chan T)
虽然语法合法(如 chan *chan int),但存在明显风险:
- 指针可能为 nil,引发 panic;
- 通道本身已是引用类型(底层指向 hchan 结构体),取地址无实际收益;
- 增加理解与维护成本,违背 Go 的简洁哲学。
// ❌ 不推荐示例(仅作对比)
func badExample(c chan *chan int) {
if chPtr := <-c; chPtr != nil {
(*chPtr) <- 100 // 需解引用,易出错
}
}? 关键要点总结
- 通道是引用类型,赋值、传参、发送时均复制其内部指针,行为等价于传指针;
- chan chan T 是标准、安全、高效的表达方式;
- 使用前确保目标通道已初始化(make(chan T)),避免向 nil 通道发送导致 goroutine 永久阻塞;
- 若需双向控制(如关闭通知),可配合 done 通道或 sync.WaitGroup 设计更健壮的通信协议。
掌握这一模式,有助于构建灵活的管道系统、动态 worker 调度器或插件化消息路由架构。










