Go select 使用深入介绍

在 Go select 教程中我们简单介绍了select的知识点。这里我们对select进行深入详细的介绍。

什么是select

select 语句用于从多个发送/接收通道操作中进行选择。 select 语句会阻塞,直到其中一个发送/接收操作准备就绪。 如果准备好多个操作,则随机选择其中一个。 语法与 switch 类似,只是每个 case 语句都是一个通道操作。 让我们通过下面的示例来更好地理解。

package main

import (

"fmt"

"time"

)

funcserver1(ch chanstring) {

time.Sleep(6 * time.Second)

ch <- "from server1"

}

funcserver2(ch chanstring) {

time.Sleep(3 * time.Second)

ch <- "from server2"

}

funcmain() {

output1 := make(chanstring)

output2 := make(chanstring)

go server1(output1)

go server2(output2)

select {

case s1 := <-output1:

fmt.Println(s1)

case s2 := <-output2:

fmt.Println(s2)

}

}

运行示例

在上面的程序中,server1 函数休眠 6 秒,然后将字符串 “from server1” 写入通道 ch。 server2 函数休眠 3 秒,然后将字符串 “from server2” 写入通道 ch。

main 函数调用 go 协程 server1 和 server2。

控制器到达 select 语句。 select 语句会阻塞,直到它的一种情况准备就绪。 在我们上面的程序中,server1 Goroutine 在 6 秒后写入 output1 通道,而 server2 在 3 秒后写入 output2 通道。 所以 select 语句会阻塞 3 秒,并等待 server2 Goroutine 写入 output2 通道。 3 秒后,程序打印如下内容

go select 并发

然后程序终止。

select 的实际应用

之所以将上述程序中的函数命名为 server1 和 server2,是为了说明select的实际使用。

假设我们有一个关键任务应用程序,我们需要尽快将输出返回给用户。 此应用程序的数据库被复制并存储在世界各地的不同服务器中。 假设函数 server1 和 server2 实际上正在与 2 个这样的服务器通信。 每个服务器的响应时间取决于每个服务器的负载和网络延迟。 我们将请求发送到两个服务器,然后使用 select 语句在相应的通道上等待响应。 首先响应的服务器由 select 选择,另一个响应被忽略。 这样我们就可以将相同的请求发送到多个服务器,并将最快的响应返回给用户:)。

Default case

当其他 case 都没有准备好时,将执行 select 语句中的 default case。 这通常用于防止 select 语句阻塞。

package main

import (

"fmt"

"time"

)

funcprocess(ch chanstring) {

time.Sleep(10500 * time.Millisecond)

ch <- "处理成功!"

}

funcmain() {

ch := make(chanstring)

go process(ch)

for {

time.Sleep(1000 * time.Millisecond)

select {

case v := <-ch:

fmt.Println("接收的值: ", v)

return

default:

fmt.Println("没有接收到值")

}

}

}

运行示例

在上面的程序中,process 函数休眠 10500 毫秒(10.5 秒),然后将字符串“处理成功”写入 ch 通道。该函数并发调用。

并发调用 process Goroutine 后,在主 Goroutine 中启动无限循环。无限循环在每次迭代开始期间休眠 1000 毫秒(1 秒),然后执行 select 操作。在前 10500 毫秒内,select 语句的第一个 case,即 case v := <-ch: 不会准备好,因为 process Goroutine 只会在 10500 毫秒后写入 ch 通道。因此在此期间将执行默认情况,并且程序将打印 10 次 “没有接收到值”。

10.5 秒后,process Goroutine 将“处理成功!”写入ch。 现在将执行 select 语句的第一个 case,程序将打印“接收的值: 处理成功!”,然后将终止。该程序将输出如下内容

Go select 的default case


死锁和 default case

package main

func main() {

ch := make(chan string) select { case <-ch: } }

<ahref='https://tools.jiyik.com/run_code/go_deadlock_default_case'class='run-code'target='_blank'>运行示例</a>

在上面的程序中,我们创建了一个通道 ch。 我们尝试在 select 中从该通道接收数据。 select 语句将永远阻塞,因为没有其他 Goroutine 在写入此通道,因此将导致死锁。 该程序将在运行时产生 panic 并显示以下内容

go 死锁select

如果存在 default case,则不会发生这样的死锁,因为 default case 将在没有其他情况准备好时执行。 使用 default case 重写上面的程序。

package main

import"fmt"

funcmain() {

ch := make(chanstring)

select {

case <-ch:

default:

fmt.Println("default case executed")

}

}

运行示例

上面程序执行结果如下(我们也可以点击运行示例在线执行)

default case executed

同样,即使 select 只有 nil 通道,也会执行 default case。

package main

import"fmt"

funcmain() {

var ch chanstring

select {

case v := <-ch:

fmt.Println("received value", v)

default:

fmt.Println("default case executed")

}

}

在上面的程序中,ch 为 nil,我们试图从select 中的 ch 读取数据。如果不存在 default case,则 select 将永远阻塞并导致死锁。 由于我们在 select 中有一个 default case,它将被执行并且程序将打印,

default case executed


随机选择 case

当 select 语句中的多个 case 准备就绪时,将随机执行其中一个。

package main

import (

"fmt"

"time"

)

funcserver1(ch chanstring) {

ch <- "from server1"

}

funcserver2(ch chanstring) {

ch <- "from server2"

}

funcmain() {

output1 := make(chanstring)

output2 := make(chanstring)

go server1(output1)

go server2(output2)

time.Sleep(1 * time.Second)

select {

case s1 := <-output1:

fmt.Println(s1)

case s2 := <-output2:

fmt.Println(s2)

}

}

运行示例

在上面的程序中,分别调用go 协程 server1 和 server2 。 然后主程序休眠 1 秒。 当控制器到达 select 语句时,server1 将 “from server1” 写入到 output1 通道,而 server2 将 “from server2” 写入到 output2 通道,因此 select 语句的两种情况都已准备好执行。 如果多次运行此程序,则输出将在 server1 或 server2 之间有所不同,具体取决于随机选择的情况。

请在本地机器上运行此程序,从而可以保证其随机性。 如果点击上面的运行示例,它执行的结果相同。


空 select

package main

funcmain() {

select {}

}

运行示例

你认为上面程序的输出是什么?

我们知道 select 语句会阻塞,直到它的一种情况被执行。 在这种情况下,select 语句没有任何 case,因此它将永远阻塞,从而导致死锁。 该程序将进入panic

go empty select 死锁

本文转载自:迹忆客(https://www.jiyik.com)

以上是 Go select 使用深入介绍 的全部内容, 来源链接: utcz.com/z/290246.html

回到顶部