golang channel
卢春风 · · 1625 次点击 · · 开始浏览golang channel 有缓冲 与 无缓冲 是有重要区别的
一个是同步的 一个是非同步的
c1:=make(chan int) 无缓冲
c2:=make(chan int,1) 有缓冲
无缓冲的 不仅仅是 向 c1 通道放 1 而是 一直要有别的携程 <-c1 接手了 这个参数,那么c1<-1才会继续下去,要不然就一直阻塞着
而 c2<-1 则不会阻塞,因为缓冲大小是1 只有当 放第二个值的时候 第一个还没被人拿走,这时候才会阻塞。
打个比喻
无缓冲的 就是一个送信人去你家门口送信 ,你不在家 他不走,你一定要接下信,他才会走。
无缓冲保证信能到你手上
有缓冲的 就是一个送信人去你家仍到你家的信箱 转身就走 ,除非你的信箱满了 他必须等信箱空下来。
有缓冲的 保证 信能进你家的邮箱
package main
import "fmt"
func Afuntion(ch chan int) {
fmt.Println("2")
<-ch } func main() { fmt.Println("1") ch := make(chan int) //无缓冲的channel go Afuntion(ch) ch <- 1 fmt.Println("3") }
执行结果:
1
2
3
package main
import "fmt"
func Afuntion(ch chan int) {
<-ch fmt.Println("2") } func main() { fmt.Println("1") ch := make(chan int) //无缓冲的channel ch <- 1 go Afuntion(ch) fmt.Println("3") }
执行结果:
1
fatal error: all goroutines are asleep - deadlock!
※(注記)从这里可以看出,对于无缓冲的channel,放入操作和取出操作不能再同一个routine中,而且应该是先确保有某个routine对它执行取出操作,然后才能在另一个routine中执行放入操作。
进一步认识golang中的并发
深入学习golang—channel
有疑问加站长微信联系(非本文作者)
入群交流(和以上内容无关):加入Go大咖交流群,或添加微信:liuxiaoyan-s 备注:入群;或加QQ群:692541889
关注微信- 请尽量让自己的回复能够对别人有帮助
- 支持 Markdown 格式, **粗体**、~~删除线~~、
`单行代码` - 支持 @ 本站用户;支持表情(输入 : 提示),见 Emoji cheat sheet
- 图片支持拖拽、截图粘贴等方式上传
收入到我管理的专栏 新建专栏
golang channel 有缓冲 与 无缓冲 是有重要区别的
一个是同步的 一个是非同步的
c1:=make(chan int) 无缓冲
c2:=make(chan int,1) 有缓冲
无缓冲的 不仅仅是 向 c1 通道放 1 而是 一直要有别的携程 <-c1 接手了 这个参数,那么c1<-1才会继续下去,要不然就一直阻塞着
而 c2<-1 则不会阻塞,因为缓冲大小是1 只有当 放第二个值的时候 第一个还没被人拿走,这时候才会阻塞。
打个比喻
无缓冲的 就是一个送信人去你家门口送信 ,你不在家 他不走,你一定要接下信,他才会走。
无缓冲保证信能到你手上
有缓冲的 就是一个送信人去你家仍到你家的信箱 转身就走 ,除非你的信箱满了 他必须等信箱空下来。
有缓冲的 保证 信能进你家的邮箱
package main
import "fmt"
func Afuntion(ch chan int) {
fmt.Println("2")
<-ch } func main() { fmt.Println("1") ch := make(chan int) //无缓冲的channel go Afuntion(ch) ch <- 1 fmt.Println("3") }
执行结果:
1
2
3
package main
import "fmt"
func Afuntion(ch chan int) {
<-ch fmt.Println("2") } func main() { fmt.Println("1") ch := make(chan int) //无缓冲的channel ch <- 1 go Afuntion(ch) fmt.Println("3") }
执行结果:
1
fatal error: all goroutines are asleep - deadlock!
※(注記)从这里可以看出,对于无缓冲的channel,放入操作和取出操作不能再同一个routine中,而且应该是先确保有某个routine对它执行取出操作,然后才能在另一个routine中执行放入操作。