Go语言 管道2

本篇文章主要介绍Go语言 管道:判断管道关闭、单向通道及多管道协同工作内容和示例。

目录

如何判断管道关闭

单向通道

Select

总结


如何判断管道关闭

需要知道一个管道的状态,如果已经关闭了,读不会有问题,会返回零;

但是如果继续写,会有崩溃风险。

可以使用map时,使用的ok-idom模式判断,如下:

map ==> v, ok := m1[0]

channel ==> v, ok := <-numChan

示例如下:

Go 复制代码
package main

import "fmt"

func main() {
   numChan := make(chan int, 10)

   // 写
   go func() {
      for i:= 0; i< 10;i++{
         numChan <- i
         fmt.Println("写入数据:", i)
      }
      close(numChan)
   }()

   for {
      v, ok := <- numChan // ok-idom模式判断
      if !ok {
         fmt.Println("管道已关闭,准备退出")
         break
      }
      fmt.Println("v:", v)
   }

   fmt.Println("结束")
}

单向通道

NumChan := make(chan int, 10) ==> 双向通道,既可以读,也可以写

单向通道:为了明确语义,一般用于函数参数

单向读通道:var numChanRead <- chan int

单向写通道:var numChanWrite chan <- int

示例如下:

Go 复制代码
package main

import (
   "fmt"
   "time"
)

func main() {
   // 单向读通道:
   // var numChanRead <- chan int
   // 单向写通道:
   // var numChanWrite chan <- int

   // 生产者消费者模型
   // c:数组+锁 thread1:写 thread2:读
   // Go: goroutine + channel

   // 1.在主函数中创建一个双向通道 numChan
   numChan := make(chan int, 10)

   // 2.将numChan 传递给producer,负责生产
   // 双向通道可以赋值给同类型的单向通道,单向不能转双向
   go producer(numChan)

   // 3.将numChan 传递给consumer 负责消费
   go consumer(numChan)

   time.Sleep(2 * time.Second)
   fmt.Println("结束")
}

// producer 生产者 ==》 提供一个只写的通道
func producer(out chan<- int) {
   for i := 0; i < 10; i++ {
      out <- i
      // data := <- out // 写通道不允许有读取操作
      fmt.Println("======> 向管道中写入数据:", i)
   }
}

// consumer 消费者 ==》 提供一个只读的通道
func consumer(in <- chan int)  {
   // in <- 1 // 读通道不允许有写入操作
   for v := range in {
      fmt.Println("从管道中读取数据:", v)
   }
}

Select

当程序中有多个channel协同工作:ch1,ch2;

某一个时刻,ch1或者ch2触发了,程序要做相应的处理,

使用select来监听多个通道,当管道被触发时(写入数据,读取数据,关闭管道)。

select语法与switch case很像,但是所有的分支条件都必须是通道io

示例如下:

Go 复制代码
package main

import (
   "fmt"
   "time"
)

func main() {
   // var ch1,ch2 chan int
   ch1 := make(chan int)
   ch2 := make(chan int)

   // 启动一个go程,负责监听两个channel
   go func() {
      for {
         fmt.Println("监听中。。。。。。。。。。。。。。。。")
         select {
         case data1 := <-ch1:
            fmt.Println("从ch1读取数据成功,data1:", data1)
         case data2 := <-ch2:
            fmt.Println("===> 从ch2读取数据成功,data2:", data2)
         default:
            fmt.Println("select default 分支 called")
            time.Sleep(time.Second)
         }
      }
   }()

   // 启动go程1 写ch1
   go func() {
      for i := 0; i < 10; i++ {
         ch1 <- i
         time.Sleep(1 * time.Second / 2)
      }
   }()

   // 启动go程2 写ch2
   go func() {
      for i := 0; i < 10; i++ {
         ch2 <- i
         time.Sleep(1 * time.Second)
      }
   }()

   for {
      fmt.Println("结束")
      time.Sleep(5 * time.Second)
   }
}

总结

本篇文章主要介绍Go语言 管道:判断管道关闭、单向通道及多管道协同工作内容和示例。

相关推荐
GW_Cheng11 分钟前
分享一个vue2的tinymce配置
开发语言·javascript·ecmascript
路人与大师20 分钟前
【Mermaid.js】从入门到精通:完美处理节点中的空格、括号和特殊字符
开发语言·javascript·信息可视化
盖世英雄酱5813627 分钟前
Read timed out问题 排查
java·数据库·后端
你怎么知道我是队长1 小时前
C语言---循环结构
c语言·开发语言·算法
BXCQ_xuan1 小时前
软件工程实践二:Spring Boot 知识回顾
java·spring boot·后端
o0o_-_1 小时前
【go/gopls/mcp】官方gopls内置mcp server使用
开发语言·后端·golang
苏三说技术2 小时前
为什么不建议在 Docker 中跑 MySQL?
后端
二饭2 小时前
Spring Boot 项目启动报错:MongoSocketOpenException 连接被拒绝排查日记
java·spring boot·后端
Dxy12393102162 小时前
python把文件从一个文件复制到另一个文件夹
开发语言·python
酷飞飞2 小时前
Qt Designer与事件处理
开发语言·qt·命令模式