3.10 Go之channel超时机制
什么是超时?
所谓的超时可以理解为token过期,需要重新操作
Go中的channel超时机制
为了防止channel被其他goroutine竞争,所以需要设置一个超时处理器
处理办法:
使用select来处理超时
Go中select的使用方法以及和switch的区别
select的特点:
-
其中有一个
case完成程序就会往下执行而不考虑其他case的情况 -
每个
case的语句里面必须是一个i/o操作 -
select块当中有多条语句可以执行(多个goroutine都不阻塞)时随机选择一条来执行
select{
case <-chan1:
// chan1成功读取到数据则执行该case
case chan2 <- 1:
// 成功向chan2写入数据则执行该case
default:
// 如果上面都没有成功,则进入default处理流程
}
所有goroutine都阻塞的情况:
-
没有定义
default则select语句会被阻塞.直到至少有一个通信可以进行下去 -
定义了
default语句,执行default.同时程序的执行会从select语句后的语句中恢复
实例代码:
package main
import (
"fmt"
"time"
)
/*
模拟一个go的超时处理办法
使用select进行处理
*/
func main() {
// 定义两个通道,一个进,一个出
ch := make(chan int)
quit := make(chan bool)
// 开启一个协程进行超时处理
go func() {
// 循环执行select
for {
select {
/* 能从ch通道当中读取到数据 */
case num := <-ch:
fmt.Println("号码 = ", num)
case <-time.After(3 * time.Second):
fmt.Println("超时")
// 同时将该条信息告知出通道--->放入成功
quit <- true
//default:
// fmt.Println("未知情况!")
// quit <- false
}
}
}()
// 循环往ch通道放值
for i := 0; i < 5; i++ {
ch <- i
time.Sleep(time.Second)
}
// 匿名接收quit通道内的值--->只是为了保证goroutine不死锁
judge := <-quit
switch judge {
case true:
fmt.Println("程序结束!")
case false:
fmt.Println("程序异常!")
}
}
分析:
-
前四次执行的是
case num的操作,最后一次执行的是case <-time.After的逻辑 -
每次往
ch通道中放入值的时候都进行了时间等待,为了超时的时候执行time.After -
**
default
It's a lonely road!!!

浙公网安备 33010602011771号