第八章 并发编程——4.select多路复用

一、引言

在Go语言的并发编程中,除了通道(Channel)这一核心概念外,select语句也扮演着至关重要的角色。select语句允许goroutine在多个通信操作中进行选择,从而实现了多路复用。本文将详细介绍select语句的用法,并通过示例帮助读者更好地理解和掌握这一知识点。

二、select语句的基本用法

select语句类似于switch语句,但它是针对通信操作而不是常规的条件判断。select语句会一直等待,直到其中一个case的通信操作可以执行。

语法结构

复制代码
select {
case <-chan1:
    // 当从chan1成功接收到数据时执行的代码
case chan2 <- 1:
    // 当成功向chan2发送数据时执行的代码
default:
    // 如果没有case可以立即执行,且存在default子句时执行的代码
}
三、select语句的详细讲解
  1. 多个case子句select语句可以包含多个case子句,每个case子句描述一个通信操作(发送或接收)。
  2. 随机选择:如果有多个case子句可以立即进行通信操作,Go语言会随机选择其中一个执行。这意味着,即使多个通道都准备好了,也无法预测哪个case会被执行。
  3. default子句:如果没有任何case子句可以立即进行通信操作,且存在default子句,则执行default代码块。这可以用于实现超时检测等功能。
四、示例

示例1:基本用法

复制代码
package main
 
import (
    "fmt"
    "time"
)
 
func main() {
    ch1 := make(chan int)
    ch2 := make(chan string)
 
    go func() {
        time.Sleep(1 * time.Second)
        ch1 <- 1
    }()
 
    go func() {
        time.Sleep(2 * time.Second)
        ch2 <- "hello"
    }()
 
    select {
    case data := <-ch1:
        fmt.Println("ch1:", data)
    case data := <-ch2:
        fmt.Println("ch2:", data)
    }
}

在这个示例中,我们创建了两个通道ch1ch2,并在两个goroutine中分别向它们发送数据。由于ch1的数据发送操作比ch2早1秒,因此select语句会选择ch1的case子句执行。

示例2:超时检测

复制代码
package main
 
import (
    "fmt"
    "time"
)
 
func main() {
    ch := make(chan int)
 
    go func() {
        time.Sleep(2 * time.Second)
        ch <- 2
    }()
 
    timeout := time.After(1 * time.Second) // 设置超时时间
 
    select {
    case data := <-ch:
        fmt.Println("Received:", data)
    case <-timeout:
        fmt.Println("Timeout")
    }
}

在这个示例中,我们创建了一个通道ch,并在一个goroutine中向它发送数据。同时,我们使用time.After函数创建了一个超时通道timeout,它在1秒后发送当前时间。由于ch的数据发送操作比超时时间晚,因此select语句会选择timeout的case子句执行,输出"Timeout"。

五、总结

select语句是Go语言并发编程中非常重要的一个特性,它允许goroutine在多个通信操作中进行选择,从而实现了多路复用。通过合理使用select语句,我们可以实现诸如超时检测、优先处理等复杂功能。

相关推荐
不老刘17 分钟前
LiveKit 本地部署全流程指南(含 HTTPS/WSS)
golang·实时音视频·livekit
Tony Bai7 小时前
再见,丑陋的 container/heap!Go 泛型堆 heap/v2 提案解析
开发语言·后端·golang
念何架构之路9 小时前
Go进阶之panic
开发语言·后端·golang
先跑起来再说9 小时前
Git 入门到实战:一篇搞懂安装、命令、远程仓库与 IDEA 集成
ide·git·后端·elasticsearch·golang·intellij-idea
Tony Bai18 小时前
“Go 2,请不要发生!”:如果 Go 变成了“缝合怪”,你还会爱它吗?
开发语言·后端·golang
灰子学技术1 天前
go response.Body.close()导致连接异常处理
开发语言·后端·golang
源代码•宸1 天前
大厂技术岗面试之谈薪资
经验分享·后端·面试·职场和发展·golang·大厂·职级水平的薪资
有代理ip3 天前
Python 与 Golang 爬虫的隐藏优势
爬虫·python·golang
天远云服3 天前
天远车辆过户查询API微服务实战:用Go语言构建高性能车况溯源系统
大数据·微服务·架构·golang
女王大人万岁3 天前
Go标准库 sync 详解
服务器·开发语言·后端·golang