当前位置: 首页 > news >正文

Go 语言中的 `select` 语句详解

select 是 Go 语言中处理通道(Channel)操作的一个强大控制结构,它允许 goroutine 同时等待多个通道操作。下面我将全面详细地解释 select 语句的各个方面。

基本语法

select 语句的基本语法如下:

select {
case <-ch1:// 如果从 ch1 成功接收数据,则执行此分支
case x := <-ch2:// 如果从 ch2 成功接收数据,则执行此分支,并将值赋给 x
case ch3 <- y:// 如果成功向 ch3 发送数据 y,则执行此分支
default:// 如果以上 case 都不满足,则执行此分支
}

工作原理

  1. 多路复用select 会阻塞,直到其中一个 case 可以执行
  2. 随机选择:当多个 case 同时就绪时,Go 会随机选择一个执行
  3. 非阻塞:当有 default 分支时,select 不会阻塞

详细特性

1. 基本通道操作

ch1 := make(chan string)
ch2 := make(chan string)go func() { ch1 <- "hello" }()
go func() { ch2 <- "world" }()select {
case msg1 := <-ch1:fmt.Println("received", msg1)
case msg2 := <-ch2:fmt.Println("received", msg2)
}

2. 超时控制

select 常与 time.After 结合实现超时:

select {
case res := <-someChan:fmt.Println(res)
case <-time.After(time.Second * 1):fmt.Println("timeout after 1 second")
}

3. 非阻塞操作

使用 default 实现非阻塞的通道操作:

select {
case msg := <-ch:fmt.Println("received", msg)
default:fmt.Println("no message received")
}

4. 永久阻塞

空的 select 会永久阻塞:

select {}
// 这常用于阻止 main 函数退出

5. 循环 select

通常与 for 循环结合使用:

for {select {case x := <-ch1:fmt.Println(x)case y := <-ch2:fmt.Println(y)case <-quit:return}
}

高级用法

1. 优先级处理

如果需要优先处理某个通道,可以这样实现:

for {select {case highPrio := <-highPriorityChan:// 处理高优先级default:select {case highPrio := <-highPriorityChan:// 处理高优先级case lowPrio := <-lowPriorityChan:// 处理低优先级}}
}

2. 动态 case

使用反射可以实现动态的 select case:

cases := []reflect.SelectCase{{Dir:  reflect.SelectRecv,Chan: reflect.ValueOf(ch1),},{Dir:  reflect.SelectRecv,Chan: reflect.ValueOf(ch2),},
}chosen, value, _ := reflect.Select(cases)
fmt.Printf("Chosen %d, value %v", chosen, value)

3. 退出模式

done := make(chan struct{})go func() {defer close(done)// 工作代码
}()select {
case <-done:// 正常完成
case <-time.After(timeout):// 超时处理
}

注意事项

  1. 死锁风险:如果所有 case 都阻塞且没有 default,会导致死锁
  2. 性能考虑:频繁的 select 可能影响性能,在高性能场景需谨慎使用
  3. 通道关闭:从已关闭的通道接收会立即返回零值,可能导致意外行为
  4. 随机选择:多个 case 就绪时的随机选择可能导致优先级问题

实际应用示例

1. 服务多个客户端

func serve(ch1, ch2 <-chan Request, quit <-chan bool) {for {select {case req := <-ch1:handleRequest(req)case req := <-ch2:handleRequest(req)case <-quit:return}}
}

2. 竞速请求

func race(url1, url2 string) (string, error) {ch := make(chan string, 2)go func() { ch <- request(url1) }()go func() { ch <- request(url2) }()select {case resp := <-ch:return resp, nilcase <-time.After(time.Second * 5):return "", fmt.Errorf("timeout")}
}

3. 工作池模式

func worker(id int, jobs <-chan int, results chan<- int) {for j := range jobs {select {case <-time.After(time.Second): // 模拟超时fmt.Printf("worker %d timeout\n", id)default:fmt.Printf("worker %d processing job %d\n", id, j)results <- j * 2}}
}

select 语句是 Go 并发编程的核心工具之一,熟练掌握它对于编写高效、健壮的并发程序至关重要。

http://www.xdnf.cn/news/176815.html

相关文章:

  • CSS元素动画篇:基于当前位置的变换动画(四)
  • 加密算法 AES、RSA、MD5、SM2 的对比分析与案例(AI)
  • (七)RestAPI 毛子(Http 缓存/乐观锁/Polly/Rate limiting)
  • 【学习笔记1】一站式大语言模型微调框架LLaMA-Factory
  • Vue2 与 Vue3 深度对比与技术解析
  • 黑马点评redis改 part 6
  • 一周学会Pandas2 Python数据处理与分析-Pandas2数据信息查看操作
  • 语音识别质量的跟踪
  • 力扣HOT100之链表:23. 合并 K 个升序链表
  • 树状数组单点操作+前缀K差分->区间K操作 -#131-#132
  • SpringBoot + SSE 实时异步流式推送
  • Linux内核中的编译时安全防护:以网络协议栈控制块校验为例
  • mAh 与 Wh:电量单位的深度解析
  • 【Pandas】pandas DataFrame rtruediv
  • 全网直播推介会,九识智能与申通快递达成全面战略合作
  • 20.压敏电阻的特性与使用注意事项
  • RuoYi-Vue项目Docker镜像构建、推送与部署完整流程
  • 云平台+MQTT+C#上位机+单片机通信
  • 在 UniApp 中实现 App 与 H5 页面的跳转及通信
  • lightrag : from lightrag.utils import EmbeddingFunc 报错
  • 04.通过OpenAPI-Swagger规范让Dify玩转Agent
  • 【Redis】set类型
  • JavaEE-多线程实战02
  • AI如何重塑CC防护行业?五大变革与实战策略解析
  • 【创新实训个人博客】multi-agent调研(2)
  • promis(resolve,reject)入门级别
  • 互联网大厂Java面试:从Spring Boot到微服务架构的实践与挑战
  • 智诚科技苏州SOLIDWORKS授权代理商的卓越之选
  • vite.config.ts 的详细配置项说明、完整代码示例及表格总结
  • 代码随想录算法训练营day12(二叉树)