[Golang] Select

[Golang] Select

文章目录

什么是select

select是Golang中一个控制结构,可以用来处理多个channel的发送和接收操作。select会阻塞,直到其中一个case可以执行,然后执行该case。如果有多个case可以执行,则随机执行一个case。

select用法

基本用法

go 复制代码
package main

func main() {
	ch1 := make(chan int,1)
	ch2 := make(chan int)
	select {
		case <-ch1:
		// ...  如果ch1读取成功,则执行该case
		case ch2<-1:
		//...  如果ch2写入成功,则执行该case
		default:
		//...  如果都不成功,执行default
	}
}

select的用法和switch比较相似,不过select的每个case表达式都是channel的读写操作,通过多个case语句

监听多个channel的读写操作是否可以执行;如果没有可以执行的case,就执行default;如果没有default,则当前goroutine会被阻塞。

空select

如果select中什么语句都没有,就会被永久阻塞。

go 复制代码
package main

func main() {
	select {
	}
}

执行结果:

golang自带的死锁检测,发现当前goroutine永久不会被唤醒,就会报死锁错误。

没有default且case永久无法执行

go 复制代码
package main

import "fmt"

func main() {
	ch1 := make(chan int, 1)
	ch2 := make(chan int, 1)
	select {
	case <-ch1:
		fmt.Println("123")
	case <-ch2:
		fmt.Println("456")
	}
}

执行结果:

两个channel中都没有数据,所以读不出数据,该goroutine会被永久阻塞,

单个case和default

go 复制代码
package main

import (
	"fmt"
	"sync"
)

func main() {
	var wg sync.WaitGroup
	wg.Add(1)
	ch1 := make(chan int, 1)
	go func() {
		defer wg.Done()
		for i := 0; i < 5; i++ {
			select {
			case <-ch1:
				fmt.Println("123")
			default:
				fmt.Println("default")
			}
		}
	}()
	wg.Wait()
}

执行结果:

多个case和default

go 复制代码
package main

import (
	"fmt"
)

func main() {
	ch1 := make(chan int, 1)
	ch2 := make(chan int, 1)

	ch1 <- 1
	ch2 <- 2

	select {
	case <-ch1:
		fmt.Println("123")
	case <-ch2:
		fmt.Println("ABC")
	default:
		fmt.Println("default")
	}

}

执行结果:

有多个case准备好时,会随机执行一个case。

IO多路复用

看到select大家都会联想到Linux中的IO多路复用模型select、poll、epoll,IO多路复用模型是为了提升程序的处理IO时间的性能,其实golang中的select和它们还是有一定区别的。

操作系统的IO多路复用,其实就可以理解为一个或少量线程处理多个IO事件。

传统的阻塞IO:可以理解为一个线程去处理一个IO事件,如果IO事件还没就绪,就阻塞等待。

IO多路复用:一个或少量线程处理多个IO事件。

golang中的select,就是用一个goroutine去监听多个channel的读写事件,提高从多个channel获取信息的效率,也可以理解为一个线程去监控多个IO事件。

相关推荐
考虑考虑7 分钟前
Redis8中新特性:TopK获取最高排名的数据
redis·后端
Chan1628 分钟前
【 Java八股文面试 | JVM篇 内存结构、类加载、垃圾回收与性能调优 】
java·jvm·spring boot·后端·spring·idea
kevinzeng39 分钟前
JdbcTemplate常用方法
后端
yunxi_051 小时前
分布式文件服务实战稿:从本地存储到对象存储的架构升级
后端·面试
百锦再1 小时前
第15章 并发编程
android·java·开发语言·python·rust·django·go
864记忆1 小时前
Qt QML 模块及其函数详解
开发语言·qt
无敌最俊朗@1 小时前
C++ 对象布局之padding(填充字节)
开发语言·c++
共享家95271 小时前
高级IO-poll
开发语言·操作系统·io
Chan161 小时前
【 Java八股文面试 | Redis篇 缓存问题、持久化、分布式锁 】
java·数据库·redis·后端·spring·缓存·面试
q***47181 小时前
Spring Boot 整合 Druid 并开启监控
java·spring boot·后端