golang的缓冲channel简单使用

BethlyRoseDaisley · · 1676 次点击 · · 开始浏览    
这是一个创建于 的文章,其中的信息可能已经有所发展或是发生改变。

# golang的缓冲channel简单使用 ---- 我们常用的是无缓冲channel : ```go make(chan type) ``` 其实make() 创建chan的第二个参数可设置缓冲channel的大小。 上述语句等价于 make(chan type, 1) 即创建了一个缓冲区大小为1channel 下面看有缓冲channel的两个例子. ### 阻塞型 demo : 协程1 :每隔1s 往有10个缓冲的channel里面写一条msg, 协程2:每隔3s 取一条msg, ```go package main import( //"fmt" "time" "strconv" log "github.com/astaxie/beego/logs" ) func main() { log.Debug("---main--- start") msgs := make(chan string, 10) i := 0 go func() { time.Sleep(3*time.Second) for { time.Sleep(1*time.Second) i++ msg := "msg " + strconv.Itoa(i) msgs <- msg log.Debug("------ put msg : ", msg) } }() go func() { for { get := <- msgs log.Debug("---------------- pop msg : ", get) time.Sleep(3*time.Second) } }() time.Sleep(100*time.Second) } ``` 可以看到当缓冲区满了以后,写channel的操作会阻塞在那里等待读端取走msg后才能写入。 ![image.png](https://static.studygolang.com/190629/30d86845fe1e4273a9adf10a51067413.png) ### 非阻塞 实际场景中我们可能不希望程序阻塞,那么可以使用select来控制,当缓冲区满了后忽略该条msg继续执行我们的程序。 ```go package main import( //"fmt" "time" "strconv" log "github.com/astaxie/beego/logs" ) func main() { log.Debug("---main--- start") msgs := make(chan string, 3) i := 0 go func() { time.Sleep(3*time.Second) for { time.Sleep(1*time.Second) i++ msg := "msg " + strconv.Itoa(i) select { case msgs <- msg: log.Debug("------ put msg : ", msg) default : log.Debug("-----msgs chan cache full sleep 1s-----") log.Debug("-----ignore this msg-----> : ", msg) } } }() go func() { for { get := <- msgs log.Debug("---------------- pop msg : ", get) time.Sleep(3*time.Second) } }() time.Sleep(100*time.Second) } ``` 可以看到,因为写端写入过快,再写入msg6时缓冲区已满,执行default丢弃了msg6,读端在取走msg5后, 取走的不是msg6,而是msg7 ![image.png](https://static.studygolang.com/190629/e20069a0a8af4cab5a4774a9d1f29204.png)

有疑问加站长微信联系(非本文作者))

入群交流(和以上内容无关):加入Go大咖交流群,或添加微信:liuxiaoyan-s 备注:入群;或加QQ群:692541889

1676 次点击  ∙  1 赞  
加入收藏 微博
暂无回复
添加一条新回复 (您需要 登录 后才能回复 没有账号 ?)
  • 请尽量让自己的回复能够对别人有帮助
  • 支持 Markdown 格式, **粗体**、~~删除线~~、`单行代码`
  • 支持 @ 本站用户;支持表情(输入 : 提示),见 Emoji cheat sheet
  • 图片支持拖拽、截图粘贴等方式上传