1.缓冲区大小不同
不带缓冲区的chan的缓冲区大小是0,带缓冲区的chan缓冲区至少是1.
2.运行方式不同
不带缓冲区的chan线程写入时会立马发生阻塞,直到有其他线程有对该chan执行接收操作且接收成功后,写入的进程才会解除阻塞。
不带缓冲区的chan线程接收时也会立马发生阻塞,直到有其他线程对该chan执行写入操作后,接收的线程才会解除阻塞。
带缓冲区的chan执行线程执行写入时
不带缓冲区
func main() {
ch:=make(chan int,0)
go func(){
time.Sleep(time.Second*3)
fmt.Println("receive over")
<-ch
}()
ch<-1
fmt.Println("send over")
}
这段代码会先3秒后才先输出 “receive over”,然后才会输出"send over",这是因为在ch<-1操作先于<-ch执行,ch<-1执行的瞬间就被block,直到三秒后<-ch执行后ch<-1操作才会unblock。
带缓冲区
ch:=make(chan int,1)
go func(){
time.Sleep(time.Second*3)
fmt.Println("receive over")
<-ch
}()
ch<-1
fmt.Println("send over")
time.Sleep(time.Second*5)
这段代码立即就会打印"send over",三秒后才会打印"receive over",这是因为带缓冲区的chan在执行发送的操作时只要缓冲区不满就会被阻塞。
总结
带缓冲区的channel:
写入阻塞条件:缓冲区满
取出阻塞条件:缓冲区没有数据
不带缓冲区的channel:
写入阻塞条件:同一时间没有另外一个线程对该chan进行读操作
取出阻塞条件:同一时间没有另外一个线程对该chan进行取操作