Go 语言 select 语句

本文介绍了Go语言中的select控制结构,类似于用于通信的switch语句,但具有随机选择特性。通过实例展示了如何使用select来实现非阻塞的通道通信,并解释了其在并发编程中的作用。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

https://hub.docker.com/

https://studygolang.com/articles/7690


select是Go中的一个控制结构,类似于用于通信的switch语句。每个case必须是一个通信操作要么是发送要么是接收

select随机执行一个可运行的case。如果没有case可运行,它将阻塞,直到有case可运行。一个默认的子句应该总是可运行的。

看介绍说,go的select与switch类似。但其实有很大不同,switch执行是顺序的,而select是随机选择:


语法

Go 编程语言中 select 语句的语法如下:

select {
    case communication clause  :
       statement(s);      
    case communication clause  :
       statement(s); 
    /* 你可以定义任意数量的 case */
    default : /* 可选 */
       statement(s);
}

以下描述了 select 语句的语法:

  • 每个case都必须是一个通信
  • 所有channel表达式都会被求值
  • 所有被发送的表达式都会被求值
  • 如果任意某个通信可以进行,它就执行;其他被忽略。
  • 如果有多个case都可以运行,Select会随机公平地选出一个执行。其他不会执行。 
    否则:
    1. 如果有default子句,则执行该语句。
    2. 如果没有default字句,select将阻塞,直到某个通信可以运行;Go不会重新对channel或值进行求值。

实例

package main

import "fmt"

func main() {
   var c1, c2, c3 chan int
   var i1, i2 int
   select {
      case i1 = <-c1:
         fmt.Printf("received ", i1, " from c1\n")
      case c2 <- i2:
         fmt.Printf("sent ", i2, " to c2\n")
      case i3, ok := (<-c3):  // same as: i3, ok := <-c3
         if ok {
            fmt.Printf("received ", i3, " from c3\n")
         } else {
            fmt.Printf("c3 is closed\n")
         }
      default:
         fmt.Printf("no communication\n")
   }    
}

以上代码执行结果为:

no communication
package main

import (
   "fmt"
)

func main() {
   c1:=make(chan int,1024)
   c2:=make(chan int,1024)

   for i:=0;i<1000;i++{
      c1<-1
      c2<-1
      fmt.Println("i===",i)
      select{
      case <-c1:
         fmt.Println("from c1")
      case <-c2:
         fmt.Println("from c2")
      }
   }
}


结果:

i=== 0
from c1
i=== 1
from c2
i=== 2
from c1
i=== 3
from c1
i=== 4
from c2
i=== 5
from c2
i=== 6
from c2
i=== 7
from c1
i=== 8
from c1
i=== 9
Go语言中,`select`语句是并发控制的关键组成部分,主要用于处理多个通信信道(channels)上的消息。它允许goroutines(协程)同时监听多个通道并根据可用的消息执行相应的分支。当`select`遇到活跃的通道时,它会选择该通道上可读的数据进行操作,否则会阻塞等待。 `select`的基本语法结构如下: ```go select { case channel1 <- value1: // 当channel1有数据可写时执行此分支 case channel2 <- value2: // 当channel2有数据可写时执行此分支 ... default: // 没有活跃通道时执行此默认分支 } ``` 这里,`case`后的`channel <- value`表示发送值到通道,如果通道已经关闭(`close(channel)`),则对应的分支会被跳过;如果没有活跃的通道,则执行`default`分支(如果有定义的话)。 `select`的应用场景包括: 1. **网络编程**:用于从多个网络套接字或连接中接收数据,或者往这些通道发送数据。 2. **异步事件驱动**:例如处理来自文件描述符、定时器、TCP连接等的I/O事件。 3. **消息传递**:在多线程或多进程间的通信中,goroutines可以通过select轮流检查多个通道是否有新的消息。 4. **信号处理**:在等待特定信号(如键盘中断或定时器到期)时使用select。 一个典型的例子可能是从多个channel中选择最早触发的消息处理: ```go package main import ( "fmt" "time" ) func main() { messageChs := make([]chan string, 3) for i := 0; i < len(messageChs); i++ { messageChs[i] = make(chan string) } var wg sync.WaitGroup wg.Add(2) go func() { defer wg.Done() time.Sleep(time.Second) // 模拟第一个消息延迟 messageChs[0] <- "Message from channel 1" }() go func() { defer wg.Done() messageChs[1] <- "Message from channel 2" }() select { case msg := <-messageChs[0]: fmt.Println("Received:", msg) case msg := <-messageChs[1]: fmt.Println("Received:", msg) default: fmt.Println("No message available") } wg.Wait() } ``` 在这个例子中,`select`会根据哪个通道先发送消息来打印结果,从而实现了并发处理。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值