Go 语言中的select语句详解及工作原理

2025-04-12 05:50

本文主要是介绍Go 语言中的select语句详解及工作原理,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

《Go语言中的select语句详解及工作原理》在Go语言中,select语句是用于处理多个通道(channel)操作的一种控制结构,它类似于switch语句,本文给大家介绍Go语言中的select语...

Go 语言中的 selectChina编程 是做什么的

在 Go 语言中,select 语句是用于处理多个通道(channel)操作的一种控制结构。它类似于 switch 语句,但专门用于并发编程,允许 Goroutine 在多个通道上等待操作(发送或接收),并在某个通道就绪时执行对应的分支。select 是 Go 并发模型中的核心特性之一,与通道和 Goroutine 紧密相关。

基本功能

select 的主要作用是:

  • 多路复用通道:同时监听多个通道的读写操作。
  • 非阻塞选择:当多个通道中有任意一个就绪时,执行对应的逻辑;如果没有通道就绪,可以执行默认分支(如果有)。
  • 并发协调:帮助 Goroutine 在不同的通信场景中协调行为。

语法

select {
case <-channel1:
    // 从 channel1 接收数据时的处理逻辑
case channel2 <- value:
    // 向 channel2 发送数据时的处理逻辑
case v := <-channel3:
    // 从 channel3 接收数据并赋值给 v 的处理逻辑
defajavascriptult:
    // 所有通道都未就绪时的默认逻辑(可选)
}
  • 每个 case 表示一个通道操作(发送或接收)。
  • default 是可选的,表示当所有通道都未就绪时执行的逻辑。

工作原理

等待通道就绪

  • select 会阻塞当前 Goroutine,直到某个 case 中的通道操作可以执行。
  • 如果多个通道同时就绪,select 会随机选择一个 case 执行(避免饥饿问题)。

非阻塞行为

  • 如果提供了 default 分支,且没有通道就绪,select 会立即执行 default 而不会阻塞。

空 select

  • 如果 select 中没有 case,会永久阻塞(类似于 for {})。

示例

示例 1:监听多个通道

package main
import (
    "fmt"
    "time"
)
func main() {
    ch1 := make(chan string)
    ch2 := make(chan string)
    go func() {
        time.Sleep(1 * time.Second)
        ch1 <- "from ch1"
    }()
    go func() {
        time.Sleep(2 * time.Second)
        ch2 <- "from ch2android"
    }()
    select {
    case msg1 := <-ch1:
        fmt.Println("Received:", msg1)
    case msg2 := <-ch2:
        fmt.Println("Received:", msg2)
    }
}
  • 输出Received: from ch1
  • 说明ch1 在 1 秒后就绪,比 ch2(2 秒)快,因此执行 ch1 的分支。

示例 2:带默认分支

package main
import (
    "fmt"
)
func main() {
    ch := make(chan string)
    select {
   编程 case msg := <-ch:
        fmt.Println("Received:", msg)
    default:
  js      fmt.Println("No message received")
    }
}
  • 输出No message received
  • ​​​​​​​说明:由于 ch 没有数据就绪,select 执行 default 分支。

示例 3:发送和接收结合

package main
import (
    "fmt"
    "time"
)
func main() {
    ch1 := make(chan string, 1)
    ch2 := make(chan string, 1)
    select {
    case ch1 <- "to ch1":
        fmt.Println("Sent to ch1")
    case msg := <-ch2:
        fmt.Println("Received from ch2:", msg)
    default:
        fmt.Println("Nothing happened")
    }
}
  • 输出Sent to ch1
  • ​​​​​​​说明ch1 是缓冲通道,可以立即发送成功,因此执行发送分支。

示例 4:超时控制

package main
import (
    "fmt"
    "time"
)
func main() {
    ch := make(chan string)
    select {
    case msg := <-ch:
        fmt.Println("Received:", msg)
    case <-time.After(2 * time.Second):
        fmt.Println("Timeout after 2 seconds")
    }
}
  • 输出Timeout after 2 seconds
  • ​​​​​​​说明time.After 创建一个定时器通道,2 秒后就绪,模拟超时逻辑。

常见用途

多路复用

在多个通道之间选择就绪的通道,避免逐一轮询。

超时处理

使用 time.After 实现操作超时。

非阻塞检查

通过 default 分支检查通道是否就绪。

协调 Goroutine

在并发任务中,根据通道状态决定下一步操作。

注意事项

随机选择

当多个 case 同时就绪时,select 随机选择一个执行,而不是按顺序。

阻塞性

没有 default 时,select 会阻塞直到某个通道就绪。

空 select

select {}

这会永久阻塞,通常用于主 Goroutine 等待。

通道关闭

如果某个通道已关闭,接收操作会立即返回零值,可能需要额外的逻辑判断。

总结

  • 是什么select 是 Go 中用于处理多通道操作的并发控制语句。
  • 做什么:监听多个通道,选择就绪的通道执行对应逻辑,支持超时和非阻塞操作。
  • 为什么用:简化并发编程,提高代码效率和可读性。

到此这篇关于Go 语言中的select是做什么的的文章就介绍到这了,更多相关Go select内容请搜索China编程(www.chinasem.cn)以前的文章或继续浏览下面的相关文章希望大家以后多多支持China编程(www.chinasem.cn)!

这篇关于Go 语言中的select语句详解及工作原理的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1154186

相关文章

Spring Security 单点登录与自动登录机制的实现原理

《SpringSecurity单点登录与自动登录机制的实现原理》本文探讨SpringSecurity实现单点登录(SSO)与自动登录机制,涵盖JWT跨系统认证、RememberMe持久化Token... 目录一、核心概念解析1.1 单点登录(SSO)1.2 自动登录(Remember Me)二、代码分析三、

SQL Server 中的 WITH (NOLOCK) 示例详解

《SQLServer中的WITH(NOLOCK)示例详解》SQLServer中的WITH(NOLOCK)是一种表提示,等同于READUNCOMMITTED隔离级别,允许查询在不获取共享锁的情... 目录SQL Server 中的 WITH (NOLOCK) 详解一、WITH (NOLOCK) 的本质二、工作

springboot自定义注解RateLimiter限流注解技术文档详解

《springboot自定义注解RateLimiter限流注解技术文档详解》文章介绍了限流技术的概念、作用及实现方式,通过SpringAOP拦截方法、缓存存储计数器,结合注解、枚举、异常类等核心组件,... 目录什么是限流系统架构核心组件详解1. 限流注解 (@RateLimiter)2. 限流类型枚举 (

Java Thread中join方法使用举例详解

《JavaThread中join方法使用举例详解》JavaThread中join()方法主要是让调用改方法的thread完成run方法里面的东西后,在执行join()方法后面的代码,这篇文章主要介绍... 目录前言1.join()方法的定义和作用2.join()方法的三个重载版本3.join()方法的工作原

Spring AI使用tool Calling和MCP的示例详解

《SpringAI使用toolCalling和MCP的示例详解》SpringAI1.0.0.M6引入ToolCalling与MCP协议,提升AI与工具交互的扩展性与标准化,支持信息检索、行动执行等... 目录深入探索 Spring AI聊天接口示例Function CallingMCPSTDIOSSE结束语

C语言进阶(预处理命令详解)

《C语言进阶(预处理命令详解)》文章讲解了宏定义规范、头文件包含方式及条件编译应用,强调带参宏需加括号避免计算错误,头文件应声明函数原型以便主函数调用,条件编译通过宏定义控制代码编译,适用于测试与模块... 目录1.宏定义1.1不带参宏1.2带参宏2.头文件的包含2.1头文件中的内容2.2工程结构3.条件编

go动态限制并发数量的实现示例

《go动态限制并发数量的实现示例》本文主要介绍了Go并发控制方法,通过带缓冲通道和第三方库实现并发数量限制,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面... 目录带有缓冲大小的通道使用第三方库其他控制并发的方法因为go从语言层面支持并发,所以面试百分百会问到

PyTorch中的词嵌入层(nn.Embedding)详解与实战应用示例

《PyTorch中的词嵌入层(nn.Embedding)详解与实战应用示例》词嵌入解决NLP维度灾难,捕捉语义关系,PyTorch的nn.Embedding模块提供灵活实现,支持参数配置、预训练及变长... 目录一、词嵌入(Word Embedding)简介为什么需要词嵌入?二、PyTorch中的nn.Em

Go语言并发之通知退出机制的实现

《Go语言并发之通知退出机制的实现》本文主要介绍了Go语言并发之通知退出机制的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录1、通知退出机制1.1 进程/main函数退出1.2 通过channel退出1.3 通过cont

Python Web框架Flask、Streamlit、FastAPI示例详解

《PythonWeb框架Flask、Streamlit、FastAPI示例详解》本文对比分析了Flask、Streamlit和FastAPI三大PythonWeb框架:Flask轻量灵活适合传统应用... 目录概述Flask详解Flask简介安装和基础配置核心概念路由和视图模板系统数据库集成实际示例Stre