当前位置: 首页 > news >正文

# Go学习-Day9

文章目录

    • 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

http://www.lryc.cn/news/151867.html

相关文章:

  • chatGPT如何在Java中使用
  • vue+axios——创建多个实例共用请求拦截器和响应拦截器(403错误信息不提示bug解决)——基础积累
  • 全球免费编程教育网站:Code.org
  • 构造函数与成员变量初始化
  • 使用Go env命令设置Go的环境
  • 目标检测YOLO实战应用案例100讲-道路场景下目标检测与分割模型的压缩研究与实现(续)
  • b站手机缓存文件转MP4
  • 一个集成的BurpSuite漏洞探测插件1.2
  • [FMMPEG] parse与 demuxer
  • 【Bug】Ubuntu 有线设置打不开无反应
  • 迈向无限可能, ATEN宏正领跑设备切换行业革命!
  • Ubuntu18.04:ORB-SLAM3使用数据集构建地图和保存点云地图
  • 找到自制电子杂志的方法了,快来看看?
  • Django请求的生命周期
  • Kotlin 中 OkHttp 使用及解析
  • 【C++代码】用栈实现队列,用队列实现栈--代码随想录
  • 肖sir__linux详解__001
  • 【Android Framework系列】第12章 RecycleView相关原理及四级缓存策略分析
  • P1886 滑动窗口 /【模板】(双端队列)+双端队列用法
  • 网络渗透day6-面试01
  • Docker 及 Docker Compose 安装指南
  • Gitlab创建一个空项目
  • C语言-内存分布(STM32内存分析)
  • Linux上配置NAT
  • springboot实现简单的消息对话
  • 「Tech初见」Linux驱动之blkdev
  • ssh配置(二、登录服务器)
  • pytorch异常——RuntimeError:Given groups=1, weight of size..., expected of...
  • 【FPGA项目】沙盘演练——基础版报文收发
  • 【C++技能树】继承概念与解析