golang学习笔记---select(3)

select 介绍

从不同并发执行的协程中,获取数据可以用select来完成。select监听的多个通道,也可以用通道发送数值。

select {
// 接收数据
case u:= <- ch1:
        ...
// 发送数据
case v:= <- ch2:
        ...
        ...
default: // no value ready to be received
        ...
}

select 基本用法

1、如果多个通道都阻塞了,会等待直到其中一个通道可以处理。
2、如果多个通道都可以处理,随机选取一个处理。
3、如果没有通道操作可以操作并且写了default语句,会执行:default(永远是可以运行的)
4、如果防止select堵塞,可以写default来确保发送不被堵塞,没有case的select就会一直堵塞。
5、当select做选择case和default操作时,case的优先级大于default。
6、select语句实现了一种监听模式,通常在无限循环中使用,通过在某种情况下,通过break退出循环

示例:
package main

import "fmt"

func main() {
	ch := make(chan int, 1)
	for i := 0; i < 10; i++ {
		select {
		case ch <- i:
		case x := <-ch:
			fmt.Println(x)
		}
	}
}

因为这个管道的缓冲值只有1,那么同一时间只会有一个case执行,这个channel不是空的就是满的。

在第一次进入循环的时候,i为0,进入到select中,开始由上向下来发现哪一个case可以执行,当计算表达式
ch <- i,也就是向管道写入数据的时候,因为这个管道现在有缓冲,那么在向管道写完数据之后,此时的case便执行完成,然后就跳出select,开始进行下一次的循环,当i=1的时候(现在这个管道里面的数据是0),再次进入select中,此时还是开始计算ch <- i 表达式,但是现在管道里面是有数据的,再次向管道中写入数据,那么会使该发送操作阻塞,此时该case便无法再执行,那么select将会继续向下执行下一个case,在下一个case中,有一个管道的接收操作x := <- ch,在这里管道里有之前第一次循环的时候放入的0这个数据,那么在这里就会将管道的数据赋值给x,从而打印出第一个数据0,那么后面的数据就和之前的过程是一样的了。

看到这里,大概就能明白select的作用了,顺便说一下,select的case语句中,都是对应一个I/O操作,准确的说是对应一个channel的I/O操作,那么到这里也应该可以理解为什么在code-1中,一个无缓冲的channel能在那段代码中产生一个deadlock

 

 

原文地址:https://www.cnblogs.com/saryli/p/13354377.html