文章目录
- Channel
- 声明
- 存入取出
- 一个简单的死锁分析
- 个人博客:CSDN博客
Channel
- Channel本质是一个队列
- 多goroutine访问时不需要加锁,Channel天然线程安全
- channel有类型,只能写入相同类型
- channel是引用类型
- channel必须初始化才能写入数据,make分配内存
声明
-
var intChan chan intintChan = make(chan int, 3)
-
java不是很熟悉,感觉chan有点像java的原子类
存入取出
-
intChan<- xxx //存入 a := <= intChan//取出
-
管道不会自然增长,不能超过容量,不能从空的管道里取出数据,会上DeadLock
-
如果想要存储任意类型的管道,可以用空借口
-
var allChan chan interface{}
-
但是,取出的时候注意类型断言
-
close(intChan)
-
channel关闭之后就不能再写入了,但是能继续读出
-
关闭之后能用for-range来遍历,如果不关闭的话会出现死锁
-
死锁的情况很多,建议多找几篇文章看看,写写实操一下
-
空的缓冲chan相当于无缓冲的chan,无缓冲的chan需要接收者,传入者,否则就会死锁,注意及时关闭
-
只向管道内写入,不读取就会deadlock,读得慢没有关系
-
关键是要给每个管道安排一个发送者,和接收者!!!
一个简单的死锁分析
package mainimport ("fmt""time"
)func write(intChan chan int) {for i := 0; i < 5; i++ {fmt.Println("写入: ", i)intChan <- itime.Sleep(time.Second)}//close(intChan)
}func read(intChan chan int, exitChan chan bool) {for {val, ok := <-intChanif !ok {break}fmt.Println("读到", val)}exitChan <- trueclose(exitChan)
}
func main() {intChan := make(chan int, 20)exitChan := make(chan bool, 1)go write(intChan)go read(intChan, exitChan)for {_, ok := <-exitChanif !ok {break}}
}
- 输出
写入: 0
读到 0
写入: 1
读到 1
写入: 2
读到 2
写入: 3
读到 3
写入: 4
读到 4
fatal error: all goroutines are asleep - deadlock!goroutine 1 [chan receive]:
main.main()E:/JetBrains/GoLandSpace/src/go_code/project01/main/hello.go:36 +0xe8goroutine 7 [chan receive]:
main.read(0x0?, 0x0?)E:/JetBrains/GoLandSpace/src/go_code/project01/main/hello.go:19 +0x99
created by main.mainE:/JetBrains/GoLandSpace/src/go_code/project01/main/hello.go:33 +0xd9Process finished with the exit code 2
- 下面是个人的分析,不一定对,有大佬可以来指正
- 如果我们不close,channel是可以读的,我们可以边读,边写,并且,读的速度是可以更慢或者更快的,go底层会通过上下文自行判断。
- 但是这里,我们写的协程,我们关闭channel,在程序运行完之后自行关闭,此时我们读的协程会卡在intChan,等待读入,但是此时还不会报错,因为协程会因为主线程结束而结束。但是后面的exitChan会导致报错
package mainimport ("fmt""time"
)func write(intChan chan int) {for i := 0; i < 5; i++ {fmt.Println("写入: ", i)intChan <- itime.Sleep(time.Second)}//close(intChan)
}func read(intChan chan int, exitChan chan bool) {for {val, ok := <-intChanif !ok {break}fmt.Println("读到", val)}fmt.Println("到了这里")//exitChan <- true//close(exitChan)
}
func main() {intChan := make(chan int, 20)exitChan := make(chan bool, 1)go write(intChan)go read(intChan, exitChan)time.Sleep(time.Second * 10)//for {// _, ok := <-exitChan// if !ok {// break// }//}
}
-
这样并没有报错,并且发现到了这里没有打印,说明read函数作为intChan的接收者一直在等待,这时候。
-
但是,主线程运行到下面的for的时候,此时exitChan是空的,因为intChan一直在死循环等待,所以触发了死锁
-
只读只写
-
var chanIn chan<- int//只写
-
var chanOut <-chan int//只读
-
select {case …}可以安全地取出数据
-
使用recover捕获协程终端 panic