读取和写通道超时, 是很多人在使用通道过程中经常遇到的, 下面就介绍一下通道,然后简单的对读写通道做超时处理
通道概念
1. 通道采用的是FIFO(先进先出)
2. 通道的创建
chn := make(chan int, 0) // 创建读写int类型的通道, 0 代表着缓存为0,即写同时必须读取, 否则将堵塞
readOnlyChn := make(<-chan int) // 只读通道
writeOnlyChn := make(chan<- int) // 只写通道
3. 关闭通道
i. 读通道直接返回0(整数类型),bytes则返回[],
ii. 写通道 会抛出异常 (send on closed channel)
4. 判断通道是否关闭
v, ok := <-chn // ok=true说明正常,ok=false说明通道关闭
5. 通道置为nil
i. 读写通道时都会死锁
ii. select 前置 nil,则select时直接忽略,select后没有其它的case则同样堵塞
示例
基本使用场景一
package main
import (
"fmt"
"time"
)
func recv(chn chan int) {
for {
select {
case v, ok := <-chn:
fmt.Println("value: ", v, "; ok: ", ok)
if !ok {
fmt.Println("channel closed!!! ")
return // 退出协程
}
default:
fmt.Println("recv alive print.")
time.Sleep(time.Second * 1)
}
}
}
func main() {
var chn = make(chan int, 0)
//var readOnlyChn = make(<-chan int)
//var writeOnlyChn = make(chan<- int)
/* 先启动接收协程, 在通道写入数据的时候进行读出, 由于之前创建了 0 缓存的通道, 写不同时读出造成堵塞 */
go recv(chn)
chn<-1024 // 写入1024数据
time.Sleep(time.Second * 2) // 延时两秒, 好让接收线程接收处理写入数据
chn = nil
//close(chn) // 关闭通道
time.Sleep(5 * time.Second)
}
上面的使用中, 当通道被置为nil的时候虽然由于加了default不会造成死锁, 但default依然会进行不断的打印执行;
有人说了, 可以在default时判断通道是否为nil, 思路是对的,但是却没那么简单, Go中只有值传递, 即你在传通道这个参数的时候是对通道引用的拷贝, 而不是其本身, 所以通道置nil,其实recv函数内部的chn还是非空的
使用场景二, 一场景升华版
这样, 不管是通道为nil还是关闭通道都能够正常监控到了, 并及时退出协程了
package main
import (
"fmt"
"time"
)
func recv(chn *chan int) {
for {
select {
case v, ok := <-*chn:
fmt.Println("value: ", v, "; ok: ", ok)
if !ok {
fmt.Println("channel closed!!! ")
return // 退出协程
}
default:
if nil == *chn { fmt.Println("channel == nil"); return }
fmt.Println("recv alive print.")
time.Sleep(time.Second * 1)
}
}
}
func main() {
var chn = make(chan int, 0)
//var readOnlyChn = make(<-chan int)
//var writeOnlyChn = make(chan<- int)
/* 先启动接收协程, 在通道写入数据的时候进行读出, 由于之前创建了 0 缓存的通道, 写不同时读出造成堵塞 */
go recv(&chn)
chn<-1024 // 写入1024数据
time.Sleep(time.Second * 2) // 延时两秒, 好让接收线程接收处理写入数据
chn = nil
//close(chn) // 关闭通道
time.Sleep(5 * time.Second)
}
写超时处理一
func main() {
var chn = make(chan int, 0) // 创建 0 缓存通道
select {
case chn<-1024: // 写数据
fmt.Println("Writed Done!!!")
case <-time.After(time.Second * 3): // 三秒超时
fmt.Println("Time out")
}
}
写超时处理(context.WithTimeout)二
func main() {
var chn = make(chan int, 0) // 创建 0 缓存通道
ctx, _ := context.WithTimeout(context.Background(), time.Second * 3)
select {
case chn<-1024: // 写数据
fmt.Println("Writed Done!!!")
case <-ctx.Done(): // 三秒超时
fmt.Println("Done: ", ctx.Err())
}
}
下一篇介绍context