select语句用于处理多个channel操作,类似switch,每个case对应一个channel的发送或接收,当多个case就绪时随机执行一个,避免饥饿。

在Go语言中,select 语句用于同时等待多个 channel 操作。它类似于 switch,但每个 case 都是一个 channel 的发送或接收操作。当多个 channel 准备就绪时,select 会随机选择一个执行,避免程序对某个 channel 产生依赖或饥饿。
基本语法与行为
select 的结构如下:
select {
case x := <-ch1:
fmt.Println("从ch1收到:", x)
case y := <-ch2:
fmt.Println("从ch2收到:", y)
case ch3 <- 100:
fmt.Println("成功发送到ch3")
default:
fmt.Println("没有就绪的channel")
}
几点关键行为:
- 如果没有 default 分支,select 会阻塞,直到至少一个 case 可以执行
- 如果有多个 case 就绪,select 随机选择一个执行
- 如果带有 default,且无 channel 就绪,立即执行 default 分支(非阻塞)
实际应用场景:并发任务结果收集
常见用途是从多个并发任务中收集结果,比如发起多个HTTP请求或后台计算:
立即学习“go语言免费学习笔记(深入)”;
ch1 := make(chan string) ch2 := make(chan string)go func() { ch1 <- fetchWebData("url1") }() go func() { ch2 <- fetchWebData("url2") }()
for i := 0; i < 2; i++ { select { case data := <-ch1: fmt.Println("收到来自ch1的数据:", data) case data := <-ch2: fmt.Println("收到来自ch2的数据:", data) } }
这样可以按任意顺序处理先完成的任务,提升响应效率。
每个应用程序都要使用数据,Android应用程序也不例外,Android使用开源的、与操作系统无关的SQL数据库--SQLite,本文介绍的就是如何为你的Android应用程序创建和操作SQLite数据库。 数据库支持每个应用程序无论大小的生命线,除非你的应用程序只处理简单的数据,那么就需要一个数据库系统存储你的结构化数据,Android使用SQLite数据库,它是一个开源的、支持多操作系统的SQL数据库,在许多领域广泛使用,如Mozilla FireFox就是使用SQLite来存储配置数据的,iPhon
配合超时机制防止阻塞
长时间等待某个 channel 可能导致程序卡住。使用 time.After 添加超时控制:
select {
case data := <-ch:
fmt.Println("正常收到数据:", data)
case <-time.After(3 * time.Second):
fmt.Println("等待超时")
}
这在网络请求、资源获取等场景中非常实用,避免无限期等待。
非阻塞读写 channel
利用 default 实现非阻塞操作,可用于尝试读取 channel 而不阻塞主流程:
select {
case msg := <-ch:
fmt.Println("立刻收到:", msg)
default:
fmt.Println("channel为空,不等待")
}
这种模式适合心跳检测、状态轮询等轻量级处理。
基本上就这些。掌握 select 的关键是理解它的“多路复用”特性,合理结合 goroutine 和 channel,能写出高效、健壮的并发程序。注意避免漏写 default 导致意外阻塞,也要小心频繁触发 default 引发忙等待。









