使用 select 切换协程

作者: freelang | 来源:发表于2018-07-31 21:29 被阅读26次

使用 select 切换协程

image

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


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可以确保发送不被阻塞!如果没有 case,select 就会一直阻塞。

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

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

代码:


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)

  }

 }

}

输出:


Received on channel 2: 5

Received on channel 2: 6

Received on channel 1: 0

Received on channel 2: 7

Received on channel 2: 8

Received on channel 2: 9

Received on channel 2: 10

Received on channel 1: 2

Received on channel 2: 11

...

Received on channel 2: 47404

Received on channel 1: 94346

Received on channel 1: 94348

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

image

相关文章

  • 使用 select 切换协程

    使用 select 切换协程 从不同的并发执行的协程中获取值可以通过关键字select来完成,它和switch控制...

  • 第14章 使用 select 切换协程

    1、切换

  • greenlet和gevent的简单使用

    greenlet模块内部使用了协程的概念,在单线程内,我们需手动调用switch切换协程,使用方法如下 执行结果 ...

  • Kotlin Primer·第七章·协程库(上篇)

    本篇只讲了协程库的使用。还有中篇讲协程的启动和切换实现原理,下篇核心讲解kotlin协程在JVM层的实现原理。这可...

  • [Android] 深入理解Kotlin协程

    Kotlin协程 协程由程序自己创建和调度,不需要操作系统调度,所以协程比线程更加轻量。相比于线程的切换,协程切换...

  • 协程 Select

    接收 发送

  • Kotlin: withContext 详解

    withContext 是 Kotlin 中一个非常有用的挂起函数,用于在协程中切换上下文。通常,我们使用协程来执...

  • 协程

    协程,又称微线程,纤程,协程是一种用户态的轻量级线程 协程是单线程 协程的好处: 1 没有上下文切换 ...

  • 协程.md

    协程协程中使用suspend修饰方法,代表该方法可在协程中挂起。但并不是协程方法必须使用suspend修饰协程和线...

  • 协程

    协程多用在函数间切换,轻量级使用资源,生成器是实现协程的基本方式,这里首先看一下专门用作协程的模块greenlet...

网友评论

    本文标题:使用 select 切换协程

    本文链接:https://www.haomeiwen.com/subject/lwrkvftx.html