【Golang】Go语言编程思想(六):Channel,第四节,Select

使用 Select

如果此时我们有多个 channel,我们想从多个 channel 接收数据,谁来的快先输出谁,此时应该怎么做呢?答案是使用 select:

go 复制代码
package main

import "fmt"

func main() {
	var c1, c2 chan int // c1 and c2 == nil
	//n1 := <-c1
	//n2 := <-c2
	// 如果我们想同时从 c1 和 c2 接收数据, 谁来的快
	// 就先要谁, 此时就需要使用 select 来帮助我们完成上述任务

	select {
	case n := <-c1:
		fmt.Println("Received from c1:", n)
	case n := <-c2:
		fmt.Println("Received from c2:", n)
	default:
		fmt.Println("No value received")
	}
}

此时程序执行将会输出 default 的内容。为 select 加上一个无限循环,将会一直输出 default 的内容,但是如果去掉 default,将会报错:

这个错误在说的是,现在没有协程启动并向 channel 当中发送数据。

现在我们新建一个生产 channel 的工厂函数 generator,在 generator 当中,我们开启一个 goroutine,在 goroutine 当中我们开启一个无限循环,每一次迭代随机 sleep 一定时间,并将不断自增的 i i i输入到 channel 当中:

go 复制代码
func generator() chan int {
	out := make(chan int)
	go func() {
		i := 0
		for {
			time.Sleep(
				time.Duration(rand.Intn(1500)) * time.Millisecond)
			out <- i
			i++
		}
	}()
	return out
}

此时修改 main 函数为:

go 复制代码
func main() {
	c1, c2 := generator(), generator()
	//n1 := <-c1
	//n2 := <-c2
	// 如果我们想同时从 c1 和 c2 接收数据, 谁来的快
	// 就先要谁, 此时就需要使用 select 来帮助我们完成上述任务
	for {
		select {
		case n := <-c1:
			fmt.Println("Received from c1:", n)
		case n := <-c2:
			fmt.Println("Received from c2:", n)
		}
	}
}

得到的结果如下:

go 复制代码
Received from c2: 0
Received from c2: 1
Received from c1: 0
Received from c1: 1
Received from c2: 2
Received from c1: 2
Received from c1: 3
Received from c2: 3
Received from c2: 4
... ...

一个更复杂的使用 select 的例子如下,下面的例子对上述代码进行了修改,加入了定时器,并使用 createWorker 和 worker 不断地开启 goroutine 并向 channel 发送数据:

go 复制代码
package main

import (
	"fmt"
	"math/rand"
	"time"
)

func generator() chan int {
	out := make(chan int)
	go func() {
		i := 0
		for {
			time.Sleep(
				time.Duration(rand.Intn(1500)) * time.Millisecond)
			out <- i
			i++
		}
	}()
	return out
}

func worker(id int, c chan int) {
	for n := range c {
		time.Sleep(time.Second)
		fmt.Printf("Worker %d received %d\n",
			id, n)
	}
}

func createWorker(id int) chan<- int {
	c := make(chan int)
	go worker(id, c)
	return c
}

func main() {
	var c1, c2 = generator(), generator()
	var worker = createWorker(0)

	var values []int
	tm := time.After(10 * time.Second)
	tick := time.Tick(time.Second)
	for {
		var activeWorker chan<- int
		var activeValue int
		if len(values) > 0 {
			activeWorker = worker
			activeValue = values[0]
		}
		select {
		case n := <-c1:
			values = append(values, n)
		case n := <-c2:
			values = append(values, n)
		case activeWorker <- activeValue:
			values = values[1:]
		case <-time.After(800 * time.Millisecond):
			// 如果每两次生成数据之间的时间超过了 800ms:
			fmt.Println("timeout")
		case <-tick:
			fmt.Println("queue len:", len(values))
		case <-tm:
			fmt.Println("bye")
			return
		}
	}
}
相关推荐
沈雅馨1 小时前
SQL语言的云计算
开发语言·后端·golang
chillxiaohan1 小时前
GO学习记录——动态创建测试http接口
学习·http·golang
小二·1 小时前
Go 语言系统编程与云原生开发实战(第2篇):并发编程深度实战 —— Goroutine、Channel 与 Context 构建高并发 API 网关
开发语言·云原生·golang
闲谈共视2 小时前
Go语言与区块链技术的渊源
开发语言·golang·区块链
csdn_aspnet3 小时前
Go语言常用算法深度解析:并发与性能的优雅实践
后端·golang·go
LOYURU3 小时前
Centos7.6安装Go
开发语言·后端·golang
小二·3 小时前
Go 语言系统编程与云原生开发实战(第1篇):从零搭建你的第一个 Go 服务 —— 理解 GOPATH、Modules 与现代 Go 工作流
开发语言·云原生·golang
Bony-12 小时前
Go语言垃圾回收机制详解与图解
开发语言·后端·golang
吴老弟i1 天前
Go 多版本管理实战指南
golang·go
Grassto1 天前
HTTP请求超时?大数据量下的网关超时问题处理方案,流式处理,附go语言实现
后端·http·golang·go