分享
  1. 首页
  2. 文章

golang通道channel的用法及特性

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

定义

通道是go语言的一种数据类型,是goroutine之间的通信机制。

初始化
var c chan TYPE
ch := make(chan int) //无缓冲通道
ch := make(chan string ,10) //有缓冲通道
ch := make(chan<- string) //只能用于接收的通道 单向通道
ch := make(<-chan float64) //只能用于发送的通道 单项通道
接收元素

从一个未初始化的通道中获取数据会永远阻塞。
从一个已关闭的通道中获取数据,如果通道中还有值可以正常获取,如果通道中没有值,会得到该通道类型的零值。
从一个没有值的通道中获取数据会被阻塞,直到通道被写入数据或者关闭。

strChan := make(chan string,3)
elem := <-strChan
elem ,ok := <-strChan
发送元素

向一个未初始化的通道中写入数据会永远阻塞。
向一个无缓冲的通道中写入数据会被终阻塞,直到该通道有至少一个接收操作进行为止,发送操作才能被唤醒。
向一个已关闭的通道写入数据会引起恐慌。所以不要在接收方去关闭一个通道。
向一个已满的通道写入数据也会被阻塞,直到有接收操作完成才能被唤醒。

strChan := make(chan string,3)
strChan<-"hello"
strChan<-"world"

用法

for与channel

for的range子句可以对通道进行读取操作

var ch = make(chan int)
for e:= range ch{
 fmt.Printf("element:%d",e)
}

for语句可以把通道已有的数据按顺序读出。当通道没有数据时,for语句会被阻塞在range子句处。

select与channel

select语句是一种仅能用于通道发送和接收的语句。一条select语句执行时会选择其中的一个分支并执行。

var intChan = make(chan int,10)
var strChan = make(chan string,10)
select{
case e:=<-intChan:
 fmt.Printf("this case wad selected,element = %v",e)
case e:=<-strChan:
 fmt.Printf("this case wad selected,element = %v",e)
 default:
 fmt.Println("default")

如果同时又多个case可以被触发,select会通过伪随机的算法随机选择一个case并执行。

timer与channel
定时器

time.Timer是time包提供的定时器。它可以在到达指定时间后发送一个元素到字段C中

timer := time.NewTimer(time.Second * 10)
fmt.Printf("now:%v",time.Now)
expiration := <-timer.C
fmt.Printf("now:%v",time.Now)

timer.C在10s之后才会有元素输出,expiration定义语句一直被阻塞,直到10s后,才能打印第二条语句。

expiration := <-time.After(time.Second * 10)

与timer是等价了,简化了timer的定义。

断续器

time.Tricker是time包提供的断续器,它能按指定时间间隔不断重复的发送元素到字段通道C中,当下一次间隔时间点到来之时,通道中还能没被取出的元素,则当前本次操作就会被立即取消。

var ch = make(chan int 3)
ticker := time.NewTricker(time.Second )
for _ = range ticker.C{
 select {
 case ch <- 1:
 case ch <- 2:
 case ch <- 3:
 }
 }

特性

并发安全

channel的数据结构如下:

type hchan struct {
 qcount uint // 数组长度,即已有元素个数
 dataqsiz uint // 数组容量,即可容纳元素个数
 buf unsafe.Pointer // 数组地址
 elemsize uint16 // 元素大小
 closed uint32
 elemtype *_type // 元素类型
 sendx uint // 下一次写下标位置
 recvx uint // 下一次读下标位置
 recvq waitq // 读等待队列
 sendq waitq // 写等待队列
 lock mutex
}

lock保证了chan的多进程之间的并发安全。

同步与异步

无缓冲通道同步
有缓冲通道异步


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

本文来自:简书

感谢作者:进击的扣子

查看原文:golang通道channel的用法及特性

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

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

用户登录

没有账号?注册
(追記) (追記ここまで)

今日阅读排行

    加载中
(追記) (追記ここまで)

一周阅读排行

    加载中

关注我

  • 扫码关注领全套学习资料 关注微信公众号
  • 加入 QQ 群:
    • 192706294(已满)
    • 731990104(已满)
    • 798786647(已满)
    • 729884609(已满)
    • 977810755(已满)
    • 815126783(已满)
    • 812540095(已满)
    • 1006366459(已满)
    • 692541889

  • 关注微信公众号
  • 加入微信群:liuxiaoyan-s,备注入群
  • 也欢迎加入知识星球 Go粉丝们(免费)

给该专栏投稿 写篇新文章

每篇文章有总共有 5 次投稿机会

收入到我管理的专栏 新建专栏