进和出都会阻塞.
先进先出队列, 出会一直阻塞到有数据, 进时当队列未满不会阻塞, 队列已满则阻塞.
read_only := make (<-chan int)
write_only := make (chan<- int)
只读或者只写一般用在参数传递中。
mutex的性能比chan高不少。
比如开源的消息队列gnatsd(NATS)就很少用chan而使用mutex,其性能是非常高的,比另一个消息队列nsq高很多。
永远是符号<-进行读取或者写入,譬如v,ok := <-c是读取,而c <- v是写入。
c := make(chan int, 1)
c <- 10 // 写入chan
v := <- c // 从chan中读取
下面的例子判断chan是否关闭:
c := make(chan int, 1)
c <- 10
v,ok := <- c // 读取,v=10,ok=true
close(c)
v,ok := <- c // 读取,v=0,ok=false
如果写不进去就丢弃,可以用select:
c := make(chan int, 1)
select {
case c <- 10: // c中放入了10,因为chan的buffer为1
default:
}
select {
case c <- 11:
default: // c中只有10,没有11
}
select {
case v,ok := <- c:
// 读出来一个,v=10, ok=true
default:
}
select {
case v,ok := <- c:
default: // 没有可读的,走这个分支
}
超时控制
select {
case <-timeout:
fmt.Println("定时任务")
case dd := <-time.After(time.Second * 3):
fmt.Println(dd, "任务超时")
}
读取时,如果没有ok,也是可以读取的。不过如果closed也是能读的,没有赋值而已;如果要知道是否closed得加ok,也就是除非chan永远不关闭,否则读取应该用v,ok := <-c而不是用v := <-c的方式。
c := make(chan int, 1)
c <- 10
close(c)
v := <- c // c=10,读取出来一个
v = <- c // c=0,实际上没有读出来,但是判断不了
c := make(chan int, 1)
c <- 10
close(c)
v,ok := <- c // c=10,ok=true,读取出来一个
v,ok = <- c // c=0,ok=false,实际上没有读出来
for-range语法可以用到通道上。循环会一直接收channel里面的数据,直到channel关闭。不同于array/slice/map上的for-range,channel的for-range只允许有一个变量。
for v:=range aChannel {
// use v
}
等价于
for {
v, ok := <- aChannel
if !ok {
break
}
// use v
}
注意,for-range对应的channel不能是只写channel。