Golang 并发 Channel的用法

news/发布时间2024/5/19 8:40:29

目录

  • Golang 并发 Channel的用法
    • 1. channel 的创建
    • 2. nil channel
      • 读写阻塞示例
      • close示例
    • 3. channel 的读写
    • 4. channel 只读只写
    • 5. 关闭channel
      • channel关闭后,剩余的数据能否取到
      • 读取关闭的channel,将获取零值
      • 使用ok判断,是否关闭
      • 使用for-range退出
      • 使用close(ch)关闭所有下游协程
    • 6. 当函数传递channel时,是传递的引用,还是值
    • 参考

Golang 并发 Channel的用法

1. channel 的创建

ch := make(chan int)

上面是创建了无缓冲的 channel,一旦有 goroutine 往 channel 发送数据,那么当前的 goroutine 会被阻塞住,直到有其他的 goroutine 消费了 channel 里的数据,才能继续运行。

ch := make(chan int, 2)

上面示例中的第二个参数表示 channel 可缓冲数据的容量。只要当前 channel 里的元素总数不大于这个可缓冲容量,则当前的 goroutine 就不会被阻塞住。

2. nil channel

nil是pointers, interfaces, maps, slices, channels 和 function 类型的零值,表示未初始化值。nil不是未定义状态,它本身就是值。error是接口类型,因此error变量可以为nil,但string不能为nil。

下面我们看下nil 通道有什么特点,空通道对操作的反应如下:

  • 从空通道读、写会永远阻塞
  • 关闭通道会终止程序(panic)

空通道是一种特殊通道,总是阻塞。对比非空已关闭的通道仍然可以进行读取,并能够读取对应类型的零值,但对于已关闭的通道发送信息会终止程序。

一般 nil channel 用在 select 上,让 select 不再从这个 channel 里读取数据

读写阻塞示例

示例如下:

func TestNil(t *testing.T) {c := make(chan int)go sendIntegers(c)addIntegers(c)
}func addIntegers(c chan int) {sum := 0t := time.NewTimer(time.Second * 5)for {select {case input := <-c:sum = sum + inputfmt.Println("addIntegers , input : " + strconv.Itoa(input) + " , sum : " + strconv.Itoa(sum))case <-t.C:c = nilfmt.Println("addIntegers , nil channel , sum : " + strconv.Itoa(sum))}}
}func sendIntegers(c chan int) {for {time.Sleep(time.Second * 1)c <- rand.Intn(100)}
}

输出如下

=== RUN   TestNil
addIntegers , input : 81 , sum : 81
addIntegers , input : 87 , sum : 168
addIntegers , input : 47 , sum : 215
addIntegers , input : 59 , sum : 274
addIntegers , nil channel , sum : 274
panic: test timed out after 30s

此示例会一直阻塞下去,addIntegers是程序的主协程会一直阻塞下去,sendIntegers是子协程同样会一直阻塞下去。

其中:输出中的panic是单元测试的Test引发的异常,不需要考虑在内。

close示例

func TestCloseNil(t *testing.T) {c := make(chan int)go writeChannel(c)num := <-cfmt.Println("main goroutine , read num : " + strconv.Itoa(num))c = nilfmt.Println("main goroutine , to close channel .")close(c)time.Sleep(time.Second * 10)}func writeChannel(c chan int) {fmt.Println("writeChannel goroutine ,  running ...")c <- 1
}

输出如下

=== RUN   TestCloseNil
writeChannel goroutine ,  running ...
main goroutine , read num : 1
main goroutine , to close channel .
--- FAIL: TestCloseNil (0.00s)
panic: close of nil channel [recovered]panic: close of nil channel

关闭nil通道会引起程序panic

3. channel 的读写

写操作

ch := make(chan int)
ch <- 1

读操作

data <- ch

当我们不再使用 channel 的时候,可以对其进行关闭:

 close(ch)

如果 channel 关闭,继续读取关闭后的 channel,不会产生 pannic,还是可以读到数据,将得到零值,即对应类型的默认值。

为了能知道当前 channel 是否被关闭,可以使用下面的写法来判断。

  • 当channel关闭时,ok = false
  • 当channel未关闭时,ok = true
 if v, ok := <-ch; !ok {fmt.Println("channel 已关闭,读取不到数据")}

另一种写法可以使用 for-range ,使用下面的写法不断的获取 channel 里的数据,直到channel关闭,跳出循环,执行后面的代码。

 for data := range ch {// get data dosomething}

4. channel 只读只写

在默认情况下,管道是双向的,可读可写,在使用 channel 时我们还可以控制 channel 只读只写操作:

声明为只写,如下:

var chan2 chan<- int
chan2 = make(chan int, 3)
chan2 <- 20

如果试着读此chan,则编译报错,编译错误如下:

invalid operation: cannot receive from send-only channel chan2 (variable of type chan<- int) compiler (InvalidReceive)

声明为只读,不可写,否则编译报错,如下:

var chan3 <-chan int
nm2 := <-chan3

函数可以声明chan只读只写,代码示例:

// 只写操作
func send(ch chan<- int, exitChan chan struct{}) {for i := 0; i < 5; i++ {time.Sleep(time.Second * 1)ch <- i}close(ch)var a struct{}exitChan <- a
}// 只读操作
func recv(ch <-chan int, exitChan chan struct{}) {for {v, ok := <-chif !ok {break}fmt.Println("recv goroutine , value : " + strconv.Itoa(v))}var a struct{}exitChan <- a
}
func TestOnlyReadWrite(t *testing.T) {ch := make(chan int, 10)exitChan := make(chan struct{}, 2)go send(ch, exitChan)go recv(ch, exitChan)var total = 0for _ = range exitChan {total++if total == 2 {break}}fmt.Println("main goroutine , 结束")
}

输出如下:

=== RUN   TestOnlyReadWrite
recv goroutine , value : 0
recv goroutine , value : 1
recv goroutine , value : 2
recv goroutine , value : 3
recv goroutine , value : 4
main goroutine , 结束
--- PASS: TestOnlyReadWrite (5.03s)

5. 关闭channel

channel关闭后,剩余的数据能否取到

golang channel关闭后,其中剩余的数据,是可以继续读取的,channel关闭之后,仍然可以从channel中读取剩余的数据,直到数据全部读取完成。

对于关闭的channel的读写需要注意两点:

  • 如果继续向channel发送数据,会引起panic,
  • 如果继续读数据,得到的是零值(对于int,就是0)。

读取关闭的channel,将获取零值

当读取已关闭的channel时,如果继续读取channel,获取到的是零值,不会堵塞,

另外即使是无缓冲的channel,也将能一直获取到零值。

代码示例如下

func TestCloseDemo01(t *testing.T) {done := make(chan struct{})ch := make(chan int, 3)ch <- 1ch <- 2ch <- 3close(ch)go func() {for {value := <-ch//此处为假设判断,value永远不会等于10if value == 10 {break}fmt.Println("read channel , value : ", value)time.Sleep(time.Second * 1)}done <- struct{}{}}()select {case <-done:fmt.Println("读取channel,正常结束")case <-time.After(time.Second * 5):fmt.Println("超时退出")}
}

输出如下:

=== RUN   TestCloseDemo01
read channel , value :  1
read channel , value :  2
read channel , value :  3
read channel , value :  0
read channel , value :  0
超时退出
--- PASS: TestCloseDemo01 (5.00s)

使用ok判断,是否关闭

读取channel,判断是否关闭:

value, ok := <-ch
  • 当channel关闭时,ok=false
  • 当channel未关闭时,ok=true

通过判断channel是否关闭,当channel关闭时,程序可以正常退出,代码示例如下:

func TestCloseDemo02(t *testing.T) {done := make(chan struct{})ch := make(chan int, 3)ch <- 1ch <- 2ch <- 3close(ch)go func() {for {value, ok := <-chif !ok {break}fmt.Println("read channel , value : ", value)time.Sleep(time.Second * 1)}done <- struct{}{}}()select {case <-done:fmt.Println("读取channel,正常结束")case <-time.After(time.Second * 5):fmt.Println("超时退出")}
}

输出如下:

=== RUN   TestCloseDemo02
read channel , value :  1
read channel , value :  2
read channel , value :  3
读取channel,正常结束
--- PASS: TestCloseDemo02 (3.03s)
PASS

使用for-range退出

for-range是使用频率很高的结构,常用它来遍历数据,range能够感知channel的关闭,当channel被发送数据的协程关闭时,range就会结束,接着退出for循环。

它在并发中的使用场景是:当协程只从1个channel读取数据,然后进行处理,处理后协程退出。

下面这个示例程序,当通道被关闭时,协程可自动退出。

func TestCloseDemo02(t *testing.T) {ch := make(chan int, 3)ch <- 1ch <- 2ch <- 3close(ch)for v := range ch {fmt.Println("value", v)}time.Sleep(time.Second * 10)
}

使用close(ch)关闭所有下游协程

  • 关闭通道,可以主动通知所有协程退出的场景

当启动100个worker时,只要main()执行关闭stopCh,每一个worker都会都到信号,进而关闭。如果main()向stopCh发送100个数据,这种就低效了。

//close关闭所有子协程
func TestCloseDemo04(t *testing.T) {ch := make(chan int, 3)stopCh := make(chan struct{})for i := 1; i < 6; i++ {worker("worker"+strconv.Itoa(i), stopCh, ch)}time.Sleep(time.Second * 5)close(stopCh)time.Sleep(time.Second * 5)
}func worker(workerName string, stopCh <-chan struct{}, ch <-chan int) {go func() {defer fmt.Println(workerName, "goroutine , worker exit")// Using stop channel explicit exitfor {select {case <-stopCh:fmt.Println(workerName, "goroutine , Recv stop signal , return")returndefault:fmt.Println(workerName, "goroutine , worker default ...")}time.Sleep(time.Second * 3)}}()
}

输出如下

=== RUN   TestCloseDemo04
worker5 goroutine , worker default ...
worker3 goroutine , worker default ...
worker4 goroutine , worker default ...
worker1 goroutine , worker default ...
worker2 goroutine , worker default ...
worker3 goroutine , worker default ...
worker2 goroutine , worker default ...
worker5 goroutine , worker default ...
worker4 goroutine , worker default ...
worker1 goroutine , worker default ...
worker4 goroutine , Recv stop signal , return
worker4 goroutine , worker exit
worker2 goroutine , Recv stop signal , return
worker2 goroutine , worker exit
worker5 goroutine , Recv stop signal , return
worker5 goroutine , worker exit
worker1 goroutine , Recv stop signal , return
worker1 goroutine , worker exit
worker3 goroutine , Recv stop signal , return
worker3 goroutine , worker exit
--- PASS: TestCloseDemo04 (10.01s)
PASS

6. 当函数传递channel时,是传递的引用,还是值

golang 传递给函数chan类型时,是值传递和引用传递?

  • golang默认都是采用值传递,即拷贝传递
  • 有些值天生就是指针(slice、map、channel)

可以看出来map和slice都是指针传递,即函数内部是可以改变参数的值的。而array是数组传递,不管函数内部如何改变参数,都是改变的拷贝值,并未对原值进行处理。

在 Go 语言中,所有的函数参数传递都是值传递(pass by value),当将参数传递给函数时,实际上是将参数的副本传递给函数。然而,这并不意味着在函数内部对参数的修改都不会影响原始数据。因为在 Go 中,有些数据类型本身就是引用类型,比如切片(slice)、映射(map)、通道(channel)、接口(interface)和指针(pointer)。当这些类型作为参数传递给函数时,虽然传递的是值,但值本身就是一个引用。

小结
Go 语言中的参数传递总是值传递,意味着传递的总是变量的副本,无论是基本数据类型还是复合数据类型。由于复合数据类型(如切片、映射、通道、接口和指针)内部包含的是对数据的引用,所以在函数内部对这些参数的修改可能会影响到原始数据。理解这一点对于编写正确和高效的Go代码至关重要。

另外即使是引用类型,比如切片,当长度或容量(比如使用 append 函数)发生变化了,可能会导致分配新的底层数组。这种情况下,原始切片不会指向新的数组,但是函数内部的切片会。因此,如果想在函数内部修改切片的长度或容量并反映到外部,应该传递一个指向切片的指针。

参考

  • https://www.cnblogs.com/-wenli/p/12350181.html
  • https://segmentfault.com/a/1190000017958702
  • https://zhuanlan.zhihu.com/p/395278270
  • https://zhuanlan.zhihu.com/p/613771870
  • Go里面如何实现广播 https://juejin.cn/post/6844903857395335182

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.bcls.cn/Ryct/3784.shtml

如若内容造成侵权/违法违规/事实不符,请联系编程老四网进行投诉反馈email:xxxxxxxx@qq.com,一经查实,立即删除!

相关文章

Arcmap excel转shp

使用excel表格转shp的时候&#xff0c;如果你的excel里面有很多字段&#xff0c;直接转很大概率会出现转换结果错误的情况&#xff0c;那么就需要精简一下字段的个数。将原来的表格文件另存一份&#xff0c;在另存为的文件中只保留关键的经度、纬度、和用于匹配的字段即可&…

芯科科技与Arduino携手推动Matter普及化

双方的合作可助力开发人员在两分钟内将新开发板配置入网 致力于以安全、智能无线连接技术&#xff0c;建立更互联世界的全球领导厂商Silicon Labs&#xff08;亦称“芯科科技”&#xff0c;NASDAQ&#xff1a;SLAB&#xff09;日前宣布&#xff0c;公司与开源硬件和软件领域的…

【Azure 架构师学习笔记】- Azure Databricks (8) --UC架构简介

本文属于【Azure 架构师学习笔记】系列。 本文属于【Azure Databricks】系列。 接上文 【Azure 架构师学习笔记】- Azure Databricks (7) --Unity Catalog(UC) 基本概念和组件 前言 UC 简单来说&#xff0c;就是管理两样东西&#xff1a;用户和元存储。 用户管理 所有Databri…

【遥感技术】ChatGPT应用指南

遥感技术主要通过卫星和飞机从远处观察和测量我们的环境&#xff0c;是理解和监测地球物理、化学和生物系统的基石。ChatGPT是由OpenAI开发的最先进的语言模型&#xff0c;在理解和生成人类语言方面表现出了非凡的能力。本文重点介绍ChatGPT在遥感中的应用&#xff0c;人工智能…

【数据结构】数组、稀疏矩阵的操作、广义表

数组 数组&#xff1a;按一定格式排列起来的&#xff0c;具有相同类型的数据元素的集合 一维数组&#xff1a;若线性表中的数据元素为非结构的简单元素&#xff0c;则称为一维数组 二维数组&#xff1a;若一维数组中的数据元素又是一维数组结构&#xff0c;则称为二维数组 …

IDEA-常用插件

1、Mybatis Log Free 当我们使用mybatis log在控制台输出sql 内容&#xff0c;输出内容将语句与参数分开打印&#xff0c;还需要手动将参数替换到指定位置。 使用对应插件后&#xff0c;自动将输出内容组装成完整的可直接执行的SQL 在插件市场 查看对应名称&#xff0c;并安装。…

alist修改密码(docker版)

rootarmbian:~# docker exec -it [docker名称] ./alist admin set abcd123456 INFO[2024-02-20 11:06:29] reading config file: data/config.json INFO[2024-02-20 11:06:29] load config from env with prefix: ALIST_ INFO[2024-02-20 11:06:29] init logrus..…

【Ubuntu】通过网线连接两台电脑以实现局域网连接的方法

有时我们需要将多台计算机连接在一起&#xff0c;以便实现数据共享、资源访问等功能。本文将介绍如何通过网线连接两台运行Ubuntu操作系统的电脑&#xff0c;以便它们能够直接通信&#xff0c;从而实现局域网连接。 1. 准备工作 在开始之前&#xff0c;请准备好&#xff1a; …

2024牛客寒假算法基础集训营4(视频讲解题目)

2024牛客寒假算法基础集训营4&#xff08;视频讲解题目&#xff09; 视频链接ABCDEFG、H&#xff08;下面是hard版本的代码两个都可以过&#xff09; 视频链接 2024牛客寒假算法基础集训营4&#xff08;视频讲解题目&#xff09; A #include<bits/stdc.h> #define en…

深入探索pdfplumber:从PDF中提取信息到实际项目应用【第94篇—pdfplumbe】

深入探索pdfplumber&#xff1a;从PDF中提取信息到实际项目应用 在数据处理和信息提取的过程中&#xff0c;PDF文档是一种常见的格式。然而&#xff0c;要从PDF中提取信息并进行进一步的分析&#xff0c;我们需要使用适当的工具。本文将介绍如何使用Python库中的pdfplumber库来…

华为OD机试真题-整数对最小和-2023年OD统一考试(C卷)-- Python3-开源

题目&#xff1a; 考察内容&#xff1a;双循环sortsum 代码&#xff1a; """ 题目分析&#xff1a; 求随机组合最小和 输入&#xff1a; 数组a个数&#xff0c; 数组元素 数组b个数&#xff0c;数组元素 对数个数输出&#xff1a; 和的最小值3 1 1 2 3 1 2 3…

时间获取、文件属性获取 2月20日学习笔记

执行两次代码&#xff0c;打印出两次执行过程中新增的文件及删除的文件 #include <sys/types.h> #include <sys/stat.h> #include <unistd.h> #include <fcntl.h> #include <stdio.h> #include <string.h> #include <dirent.h>#def…

Docker容器实战

"爱在&#xff0c;地图上&#xff0c;剥落~" Mysql 容器化安装 我们可以在 docker hub上&#xff0c;进入mysql的镜像仓库&#xff0c;找到适合的版本。 直接拉取镜像: docker pull mysql:latest 我们知道 msyql 的默认端口是 3306 &#xff0c;而且有密码&#x…

MYSQL-入门

一.安装和连接 1.1 安装 mysql安装教程&#xff1a; 2021MySql-8.0.26安装详细教程&#xff08;保姆级&#xff09;_2021mysql-8.0.26安装详细教程(保姆级)_mysql8.0.26_ylb呀的博客-cs-CSDN博客 workbench安装&#xff1a; MySQL Workbench 安装及使用-CSDN博客 1.2 配…

强大的文本绘图——PlantUML

PlantUML是一款开源工具&#xff0c;它允许用户通过简单的文本描述来创建UML图&#xff08;统一建模语言图&#xff09;。这种方法可以快速地绘制类图、用例图、序列图、状态图、活动图、组件图和部署图等UML图表。PlantUML使用一种领域特定语言&#xff08;DSL&#xff09;&am…

本机防攻击简介

定义 在网络中&#xff0c;存在着大量针对CPU&#xff08;Central Processing Unit&#xff09;的恶意攻击报文以及需要正常上送CPU的各类报文。针对CPU的恶意攻击报文会导致CPU长时间繁忙的处理攻击报文&#xff0c;从而引发其他业务的中断甚至系统的中断&#xff1b;大量正常…

idea如何在一个service窗口中显示多个服务教程

idea在service窗口中显示多个服务 展示效果如下: 找到.idea > workspace.xml 中找到 RunDashboard 替换成如下 <component name"RunDashboard"><option name"configurationTypes"><set><option value"SpringBootApplicatio…

反序列化字符串逃逸 [安洵杯 2019]easy_serialize_php1

打开题目 $_SESSION是访客与整个网站交互过程中一直存在的公有变量 然后看extract()函数的功能&#xff1a; extract($_POST)就是将post的内容作为这个函数的参数。 extract() 函数从数组中将变量导入到当前的符号表(本题的作用是将_SESSION的两个函数变为post传参) function…

【鸿蒙 HarmonyOS 4.0】状态管理

一、介绍 资料来自官网&#xff1a;文档中心 在声明式UI编程框架中&#xff0c;UI是程序状态的运行结果&#xff0c;用户构建了一个UI模型&#xff0c;其中应用的运行时的状态是参数。当参数改变时&#xff0c;UI作为返回结果&#xff0c;也将进行对应的改变。这些运行时的状…

氢气传感器与氢冷发电机:氢能应用中的关键技术及其安全监测

​ ​随着全球对可再生能源的迫切需求&#xff0c;氢能作为一种清洁、高效的能源形式&#xff0c;正逐渐受到人们的青睐。在氢能利用的过程中&#xff0c;氢气传感器和氢冷发电机成为了不可或缺的关键技术。然而&#xff0c;氢气作为一种易燃易爆的气体&#xff0c;其安全使…
推荐文章