<<GO语言核心36讲笔记-Part10>> - Go语言中文社区

<<GO语言核心36讲笔记-Part10>>


通道的基本操作#


通道本身是并发安全的,这也是go语言自带的唯一一个可以满足并发安全的的类型。

通道的初始化

通道的创建和字典一样,用的也是内置的make函数,make函数后面的参数代表了chan的长度,是否是带缓冲的chan。

ch1 := make(chan int, 3)//创建一个int类型的chan,带3个缓冲

通道的特性

1.对于同一个通道,发送操作之间是互斥的,接收操作之间也是互斥的。

2.发送操作和接收操作中对元素的操作都是不可分割的

3.发送操作和接收操作在完全完成之前都会被阻塞。

对于第一个特性来说,指的是接收和发送的时候都只允许一个用户操作。就算是不同的goroutine,也只允许一个用户操作。

一定要注意,往通道里面写数据的时候,写的是元素的副本,并不是元素的本身。读取的时候,也是先生成在通道里面的副本,然后准备给接收方,删除在通道的元素。

第三个特性结合第二个特性,指的就是在通道读写的过程中必须完成,否则一直处于阻塞。
写的时候两个操作:“复制元素值”,“放置副本到通道内部” 这两个步骤必须完成,否则会阻塞所在的goroutine。
读的时候三个步骤:“复制通道内的元素值“,”复制副本到接收方“,”删掉原值”也是要阻塞的。


发送和接收操作什么时候会被长时间阻塞

缓冲通道

对于缓冲通道来说,如果通道满了,那么所有的写操作会依次进入等待队列。等到有读操作后按照队列顺序依次执行对应的goroutine。读操作的时候也是同样的原理,会一直等待有数据的写入。

非缓冲通道

对于非缓冲通道而言,不断是数据的读取还是写入,从一开始就是阻塞的了,直到配对的操作。非缓冲通道,是直接从发送方复制到接收方,不会用非缓冲通道做中转。相当于是同步传输,而缓冲通道是异步传输。

对于nil类型的通道,不管他是什么类型。读写操作都会一直阻塞。所以一定要make初始化!!!!!!!


发送操作和接收操作什么时候会触发panic?

1.对已经初始化,但未关闭的通道来说,收发操作一定不会引发panic
2.通道已经关闭,写操作会触发panic
3.读操作,可以返回两个值,一个是数据,一个是bool类型的代表通道是否关闭的值,如果通道关闭为false,否则为true。但记住通过第二个判断是否关闭是有延迟的(可能有数据还没读出,但对方close了chan,这种情况下,为true,同时返回的是通道里的数据)

思考题

1.通道长度代表着什么,什么时候喝容量相同

长度代表的是当前通道包含的元素个数,容量的话就是make时候的值

2.元素值在经过通道传递时会被复制,那么这个是浅表复制还是深层复制?

浅表复制

func main() {
    ch1 := make(chan int, 2)
    ch2 := make(chan []int, 2)
    l := []int{1,2,3}
    var l2 []int
    a := 0
    var b int
    go func() {
        ch1 <- a
        ch2 <- l
    }()

    select {
        case b = <-ch1:
            b = 100
    }
    select {
    case l2 = <-ch2:
        l2[0] = 100
    }
    fmt.Println(b,a)/100,0 切记这个地方不是发生了深层复制,而是go本身特性,对于基本数据类型,使用的是copy操作,所以内存不同。
    fmt.Println(l[0],l2[0]) //100,100

}
版权声明:本文来源简书,感谢博主原创文章,遵循 CC 4.0 by-sa 版权协议,转载请附上原文出处链接和本声明。
原文链接:https://www.jianshu.com/p/22af1f89480a
站方申明:本站部分内容来自社区用户分享,若涉及侵权,请联系站方删除。
  • 发表于 2020-01-12 12:58:11
  • 阅读 ( 708 )
  • 分类:Go

0 条评论

请先 登录 后评论

官方社群

GO教程

猜你喜欢