脚本之家,脚本语言编程技术及教程分享平台!
分类导航

Python|VBS|Ruby|Lua|perl|VBA|Golang|PowerShell|Erlang|autoit|Dos|bat|

服务器之家 - 脚本之家 - Golang - Go使用select切换协程入门详解

Go使用select切换协程入门详解

2022-08-20 14:11宇宙之一粟 Golang

这篇文章主要为大家介绍了Go使用select切换协程入门详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪

前言

在 Go 中,可以通过关键字 select 来完成从不同的并发执行的协程中获取值,它和 switch 控制语句非常相似,也被称作通信开关;它的行为像是“你准备好了吗”的轮询机制;

select 监听进入通道的数据,也可以是用通道发送值的时候。

select 是 Go 在语言层面提供的多路 I/O 复用机制,用于检测多个管道是否就绪(即可读或可写),其特性与管道息息相关。

语法格式:

?
1
2
3
4
5
6
7
8
9
select {
case u:= <- ch1:
        ...
case v:= <- ch2:
        ...
        ...
default: // no value ready to be received
        ...
}

default 语句是可选的;fallthrough 行为,和普通的 switch 相似,是不允许的。在任何一个 case 中执行 break 或者 return,select 就结束了。

select 做的就是:选择处理列出的多个通信情况中的一个。

  • 如果都阻塞了,会等待直到其中一个可以处理
  • 如果多个可以处理,随机选择一个
  • 如果没有通道操作可以处理并且写了default 语句,它就会执行:default 永远是可运行的(这就是准备好了,可以执行)。

select 中使用发送操作并且有 default 可以确保发送不被阻塞!如果没有 defaultselect 就会一直阻塞。default 不能处理管道读写操作,

select 语句实现了一种监听模式,通常用在(无限)循环中;在某种情况下,通过 break 语句使循环退出。

程序示例

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
package main
import (
    "fmt"
    "time"
)
func main() {
    ch1 := make(chan int)
    ch2 := make(chan int)
    go pump1(ch1)
    go pump2(ch2)
    go suck(ch1, ch2)
    time.Sleep(1e9)
}
func pump1(ch chan int) {
    for i := 0; ; i++ {
        ch <- i * 2
    }
}
func pump2(ch chan int) {
    for i := 0; ; i++ {
        ch <- i + 5
    }
}
func suck(ch1, ch2 chan int) {
    for {
        select {
        case v := <-ch1:
            fmt.Printf("Received on channel 1: %d\n", v)
        case v := <-ch2:
            fmt.Printf("Received on channel 2: %d\n", v)
        }
    }
}

在程序 goroutine_select.go 中有 2 个通道 ch1ch2

三个协程 pump1()pump2()suck()

这是一个典型的生产者消费者模式。在无限循环中,ch1ch2 通过 pump1()pump2() 填充整数;suck() 也是在无限循环中轮询输入的,通过 select 语句获取 ch1ch2 的整数并输出。选择哪一个 case 取决于哪一个通道收到了信息。程序在 main 执行 1 秒后结束。

运行结果:

Received on channel 2: 148120
Received on channel 2: 148121
Received on channel 2: 148122
Received on channel 2: 148123
Received on channel 2: 148124
Received on channel 2: 148125
Received on channel 2: 148126
Received on channel 1: 296784
Received on channel 2: 148127
Received on channel 2: 148128
Received on channel 2: 148129
Received on channel 1: 296786
Received on channel 1: 296788

一秒内的输出非常惊人,如果我们给它计数(goroutine_select2.go),得到了 296788 个左右的数字。

select 特性预览

管道读写

select 只能作用于管道,包括数据的读取和写入。例如:

?
1
2
3
4
5
6
7
8
9
10
11
12
package main
import "fmt"
func selectDemo(c chan string) {
    recv := ""
    send := "Hello"
    select {
    case recv = &lt;-c:
        fmt.Printf("Received %s\n", recv)
    case c &lt;- send:
        fmt.Printf("Sent %s\n", send)
    }
}
  • 如果管道中没有缓存,如下:
?
1
2
3
4
func main() {
    c := make(chan string)
    selectDemo(c)
}

此时管道既不能读也不能写,两个 case 语句都不执行,select 陷入阻塞

  • 如果管道中有缓冲区且还可以存放至少一个数据,如下:
?
1
2
3
4
func main() {
    c := make(chan string, 1)
    selectDemo(c)
}

此时,管道可以写入,写操作对应的 case 语句得到执行,且执行结束后函数退出。

Go使用select切换协程入门详解

  • 如果管道有缓冲区且缓冲区中已放满数据,如下:
?
1
2
3
4
5
func main() {
    c := make(chan string, 1)
    c <- "你好,向你说再见!"
    selectDemo(c)
}

Go使用select切换协程入门详解

此时,管道可以读取,读操作对应的 case 语句得到执行,且执行结束后函数退出。

  • 管道有缓冲区,缓冲区中已有部分数据还可以存入数据,如下:
?
1
2
3
4
5
func main() {
    c := make(chan string, 2)
    c &lt;- "你好,向你说再见!"
    selectDemo(c)
}

管道的缓冲区有部分且还可以存入数据,此时管道既可以读取也可以写入,select 将选取一个 case 语句执行,任意一个 case 语句执行结束后函数就退出。

Go使用select切换协程入门详解

总结

select 的每个 case 语句只能操作一个管道,要么写入数据,要么读取数据;

如果管道中没有数据读取操作则会阻塞,如果管道中没有空余的缓冲区则写入操作会阻塞;

当 select 的多个 case 语句中的管道均阻塞时,整个 select 语句也会陷入阻塞,直到任意一个管道解除阻塞;

如果多个 case 语句均没有阻塞,那么 select 将随机挑选一个 case 执行。

以上就是Go使用select切换协程入门详解的详细内容,更多关于Go select 切换协程的资料请关注服务器之家其它相关文章!

原文链接:https://juejin.cn/post/7133237032452620324

延伸 · 阅读

精彩推荐
  • Golang一步步教你编写可测试的Go语言代码

    一步步教你编写可测试的Go语言代码

    相信每位编程开发者们应该都知道,Golang作为一门标榜工程化的语言,提供了非常简便、实用的编写单元测试的能力。本文通过Golang源码包中的用法,来学...

    tabalt4412020-05-04
  • Golanggolang网络通信超时设置方式

    golang网络通信超时设置方式

    这篇文章主要介绍了golang网络通信超时设置方式,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧...

    翔云12345611002021-02-25
  • GolangGolang科学计数法转换string数字输出的实现

    Golang科学计数法转换string数字输出的实现

    最近接手一个商城运单号模块,接手后发现有部分运单号返回给前端是按照科学计数法的方式返回,本文就介绍一下Golang科学计数法转换string数字输出,感...

    孙龙-程序员10222021-08-14
  • GolangGo语言非main包编译为静态库并使用的示例代码

    Go语言非main包编译为静态库并使用的示例代码

    本文以Windows为例,介绍一下如何将Go的非main包编译为静态库,用户又将如何使用。通过实际项目创建常规工程,通过示例代码给大家介绍的非常详细,需要...

    witton12212021-08-14
  • GolangGolang中如何实现枚举详析

    Golang中如何实现枚举详析

    举就是将数据值一一列出来,枚举可以用来表示一些固定的值,枚举是常量组成的,下面这篇文章主要给大家介绍了关于Golang中如何实现枚举的相关资料,需要的...

    峰子20127742022-07-11
  • Golanggo语言之给定英语文章统计单词数量(go语言小练习)

    go语言之给定英语文章统计单词数量(go语言小练习)

    这篇文章给大家分享go语言小练习给定英语文章统计单词数量,实现思路大概是利用go语言的map类型,以每个单词作为关键字存储数量信息,本文通过实例代...

    传参2422020-06-02
  • Golang模块一 GO语言基础知识-库源码文件

    模块一 GO语言基础知识-库源码文件

    这篇文章主要介绍了模块一 GO语言基础知识-库源码文件,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考...

    lvp10512021-03-27
  • GolangGolang报“import cycle not allowed”错误的2种解决方法

    Golang报“import cycle not allowed”错误的2种解决方法

    这篇文章主要给大家介绍了关于Golang报"import cycle not allowed"错误的2种解决方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考...

    AlbertGou20912020-05-17