第八章 并发编程——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语句,我们可以实现诸如超时检测、优先处理等复杂功能。

相关推荐
Chandler245 小时前
Go:反射
开发语言·后端·golang
盒子69105 小时前
go for 闭环问题【踩坑记录】
开发语言·后端·golang
techdashen8 小时前
性能比拼: Rust vs Zig vs Go
开发语言·golang·rust
{⌐■_■}13 小时前
【go】什么是Go语言中的GC,作用是什么?调优,sync.Pool优化,逃逸分析演示
java·开发语言·javascript·jvm·数据库·后端·golang
_yingty_14 小时前
GO语言入门:常用数学函数2
java·学习·算法·golang
虽千万人 吾往矣16 小时前
golang channel源码
开发语言·后端·golang
RationalDysaniaer1 天前
gorm基础:自定义数据类型
golang
web守墓人1 天前
【gpt生成-其一】以go语言为例,详细描述一下 :语法规范BNF/EBNF形式化描述
前端·gpt·golang
yoke菜籽1 天前
k8s报错kubelet.go:2461] “Error getting node“ err=“node \“k8s-master\“ not found“
golang·kubernetes·kubelet