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

Golang通道(Channel)原理解析

引言

并发编程是现代软件开发中的一个重要主题。Golang作为一门并发友好的编程语言,提供了一种简单而强大的机制,即通道(Channel),用于在不同的Goroutine之间进行通信和同步。通道的设计和原理是Golang并发模型的核心概念之一,本文将深入探讨Golang通道的原理,包括概念、用法、场景和案例。

概念

通道是Golang中用于在不同的Goroutine之间进行通信和同步的特殊类型。通道可以用于发送和接收数据,确保数据安全传输,并防止并发访问数据的竞态条件。通道基于CSP(Communicating Sequential Processes)模型,通过使用通道来实现Goroutine之间的消息传递和同步。

通道具有以下特点:

  • 通道是一种类型,可以通过使用make函数来创建。例如,创建一个整数类型的通道:ch := make(chan int)
  • 通道可以被用于发送和接收数据。发送操作使用<-运算符:ch <- data,接收操作使用变量来接收数据:data := <- ch
  • 通道是阻塞的,当发送数据到通道时,如果通道已满,则发送操作将被阻塞,直到有其他Goroutine从通道中接收数据。同样,当从通道接收数据时,如果通道为空,则接收操作将被阻塞,直到有其他Goroutine向通道发送数据。
  • 通道可以设定缓冲区大小,用于控制通道的容量。在创建通道时,可以指定缓冲区大小:ch := make(chan int, bufferSize)。缓冲区大小决定了通道可以存储的数据量,当通道满时发送操作将被阻塞,当通道空时接收操作将被阻塞。
  • 通道可以用于进行同步操作。在接收操作前,如果通道中没有数据可用,接收操作将被阻塞,直到有其他Goroutine向通道发送数据。在发送操作前,如果通道已满,发送操作将被阻塞,直到有其他Goroutine从通道中接收数据。
  • 通道是类型安全的,只能发送和接收与通道声明的类型相同的数据。

用法

通道的使用非常简单和直观。首先,我们需要创建一个通道,可以使用make函数来创建一个通道实例。例如,创建一个字符串类型的通道:

ch := make(chan string)

接下来,我们可以在不同的Goroutine之间进行数据的发送和接收操作。发送操作使用<-运算符,接收操作使用变量来接收数据。例如,发送一个字符串到通道:

ch <- "Hello, Channel!"

接收数据时,我们可以使用变量来接收通道中的数据。例如,从通道中接收一个字符串:

data := <-ch
fmt.Println(data) // 输出: Hello, Channel!

以上就是通道的基本用法。通道的发送和接收操作都是阻塞的,这意味着在发送或接收数据时,如果条件不满足,操作将被阻塞,直到条件满足为止。

场景

通道可以用于多种并发场景,包括数据传递、同步和信号量等。以下是几个常见的场景示例:

数据传递

通道可以用于在不同的Goroutine之间传递数据。例如,一个Goroutine生成数据,另一个Goroutine处理数据。通过使用通道,可以确保数据安全地传递,并且不需要使用额外的锁机制。

func producer(ch chan<- int) {for i := 0; i < 10; i++ {ch <- i // 发送数据到通道}close(ch) // 关闭通道
}func consumer(ch <-chan int) {for num := range ch {fmt.Println(num) // 输出接收到的数据}
}func main() {ch := make(chan int)go producer(ch)consumer(ch)
}

在上面的例子中,producer函数向通道发送一系列整数,consumer函数从通道接收这些整数并进行处理。通过通道的使用,我们可以在两个Goroutine之间安全地传递数据。

同步

通道可以用于实现Goroutine之间的同步。通过使用通道,我们可以确保某个操作在其他Goroutine完成之前不会执行,从而实现同步。下面是一个使用通道实现同步的示例:

func worker(ch chan bool) {// 执行一些任务time.Sleep(time.Second * 5)ch <- true // 任务完成,发送信号到通道
}func main() {ch := make(chan bool)go worker(ch)<-ch // 等待接收信号,阻塞当前Goroutinefmt.Println("Task completed!")
}

在上面的例子中,worker函数执行一些长时间的任务,任务完成后向通道发送一个布尔值信号。main函数在启动workerGoroutine后,会阻塞在<-ch操作,直到接收到信号,才会继续执行后面的代码。

信号量

通道还可以用作信号量,用于限制某个资源的并发访问数量。通过创建一个带有缓冲区大小的通道,并在需要访问资源时获取通道中的元素,可以实现对资源的并发访问控制。

func worker(ch chan bool, id int) {<-ch // 获取通道中的元素,表示占用一个资源fmt.Println("Worker", id, "start working...")time.Sleep(time.Second * 2)ch <- true // 释放资源,发送信号到通道fmt.Println("Worker", id, "end working...")
}func main() {ch := make(chan bool, 3) // 创建带有3个资源的通道for i := 1; i <= 5; i++ {go worker(ch, i)}time.Sleep(time.Second * 5)
}

在上面的例子中,我们创建了一个带有3个资源的通道。通过在worker函数中获取和释放通道中的元素,我们限制了并发访问资源的数量为3个。这样可以确保同一时间只有3个Goroutine可以访问资源,其他的Goroutine需要等待直到有资源可用。

案例

让我们通过一个完整的案例来演示通道的使用。假设我们有一个计算密集型的任务,我们想要将其拆分成多个小任务,并使用多个Goroutine并行执行。通过使用通道来传递数据和收集结果,我们可以高效地完成整个任务。

func worker(tasks <-chan int, results chan<- int) {for task := range tasks {// 执行计算密集型任务result := task * 2results <- result // 将结果发送到通道}
}func main() {numTasks := 100numWorkers := 10tasks := make(chan int)results := make(chan int)// 启动多个worker Goroutinefor i := 0; i < numWorkers; i++ {go worker(tasks, results)}// 发送任务到通道for i := 0; i < numTasks; i++ {tasks <- i}close(tasks) // 关闭任务通道// 收集结果for i := 0; i < numTasks; i++ {result := <-resultsfmt.Println("Result:", result)}
}

此案例中,我们通过创建一个worker函数来执行计算密集型任务。该函数从tasks通道接收任务,并将结果发送到results通道。

在main函数中,我们创建了两个通道:tasks用于发送任务,results用于接收结果。我们还定义了numTasks和numWorkers来表示任务数量和worker数量。

接下来,我们使用for循环启动了numWorkers个worker Goroutine。每个worker Goroutine都会从tasks通道接收任务,并执行计算密集型任务,将结果发送到results通道。

然后,我们使用for循环将numTasks个任务发送到tasks通道中。之后,我们关闭了tasks通道,表示任务发送完毕。

最后,我们使用for循环从results通道中接收结果,并打印出来。

通过使用通道来传递数据和收集结果,我们实现了任务的并行执行。每个worker Goroutine都独立地执行任务,并将结果发送到results通道中。在主函数中,我们通过从results通道中接收结果来收集最终的计算结果。

这种并行执行的方式可以提高计算密集型任务的执行效率,同时也可以更好地利用多核处理器的性能。

相关文章:

Golang通道(Channel)原理解析

引言 并发编程是现代软件开发中的一个重要主题。Golang作为一门并发友好的编程语言&#xff0c;提供了一种简单而强大的机制&#xff0c;即通道&#xff08;Channel&#xff09;&#xff0c;用于在不同的Goroutine之间进行通信和同步。通道的设计和原理是Golang并发模型的核心…...

使用树莓派搭建文件共享服务器-samba服务器

局域网内部通过文件共享来传输文件是一种非常方便的方式&#xff0c;小米摄像头也支持用文件共享smb模式将视频备份到局域网中的文件服务器上。之前我一直使用荣耀pro路由器游戏版&#xff0c;是自带USB接口支持文件共享服务的&#xff0c;接上USB移动硬盘&#xff0c;小米摄像…...

GitLab使用webhook触发Jenkins自动构建

1、jenkins安装gitlab插件 在插件管理中&#xff0c;搜索gitlab安装这个插件。 2、job中配置webhook地址和密钥 进入job设置&#xff0c;构建触发器中就可以看到gitlab的webhook配置&#xff0c;复制URL地址和随机令牌至gitlab中 勾选后&#xff0c;就可以展开设置&#xff…...

柔性数组的使用及注意事项

1.柔性数组在结构体当中,并且在结构体的最后面. 2.结构体中除了柔型数组外至少还要有一个其他成员. 3.sizeof()返回结构体的大小不包含柔性数组的大小. 4.malloc 例:struct sdshdr16 *p malloc(sizeof (struct sdshdr16) 32); // 32 为柔性数组的大小 5.free 例: fre…...

数学建模——最优连接(基于最小支撑树)

一、概念 1、图的生成树 由图G(V,E)的生成子图G1(V,E1)(E1是E的子集&#xff09;是一棵树&#xff0c;则称该树为图G的生成树&#xff08;支撑树&#xff09;&#xff0c;简称G的树。图G有支撑树的充分必要条件为图G连通。 2、最小生成树问题 连通图G(V,E)&#xff0c;每条边…...

【LeetCode】43. 字符串相乘

1 问题 给定两个以字符串形式表示的非负整数 num1 和 num2&#xff0c;返回 num1 和 num2 的乘积&#xff0c;它们的乘积也表示为字符串形式。 注意&#xff1a;不能使用任何内置的 BigInteger 库或直接将输入转换为整数。 示例 1: 输入: num1 “2”, num2 “3” 输出: “…...

车联网之电子围栏模型使用翻滚窗口【二十一】

1. 电子围栏模型使用翻滚窗口 l 创建90秒翻滚窗口,计算中电子围栏信息(ElectricFenceModel中的值根据车辆是否在围栏内进行设置) 设置电子围栏水印 根据vin进行分组 创建翻滚窗口,90秒为一个窗口周期 自定义窗口函数,计算电子围栏中和围栏外车辆信息 1.1 创建90秒翻滚窗口 …...

图形库篇 | EasyX | 基本介绍

图形库篇 | EasyX | 基本介绍 简介 EasyX 是一个针对 C/C 的图形库&#xff0c;可以帮助 C/C 初学者快速上手图形和游戏编程。 比如&#xff0c;可以基于 EasyX 图形库很快的用几何图形画一个房子&#xff0c;或者一辆移动的小车&#xff0c;可以编写俄罗斯方块、贪吃蛇、黑白…...

凉鞋的 Unity 笔记 202. 变量概述与简介

202. 变量概述与简介 想要用好变量不是一件简单的事情&#xff0c;因为变量需要命名。 我们可以从两个角度看待一个变量&#xff0c;第一个角度是变量的功能&#xff0c;第二个是变量的可读性。 变量的功能其实非常简单&#xff0c;变量可以存储一个值&#xff0c;这个值是特…...

基于主动移频法与AFD孤岛检测的单相并网逆变器仿真(Simulink仿真实现)

&#x1f4a5;&#x1f4a5;&#x1f49e;&#x1f49e;欢迎来到本博客❤️❤️&#x1f4a5;&#x1f4a5; &#x1f3c6;博主优势&#xff1a;&#x1f31e;&#x1f31e;&#x1f31e;博客内容尽量做到思维缜密&#xff0c;逻辑清晰&#xff0c;为了方便读者。 ⛳️座右铭&a…...

jvm的jshell,学生的工具

jshell 在我眼里&#xff0c;只能作为学校教学的一个玩具&#xff0c;事实上官方也做了解释&#xff0c;以下是官方的解释&#xff1a; 在学习编程语言时&#xff0c;即时反馈很重要&#xff0c;并且 它的 API。学校引用远离Java的首要原因 教学语言是其他语言有一个“REPL”…...

深度解析 Bing 搜索引擎的排名因素与算法

自从和ChatGPT搞上了&#xff0c;Bing在搜索引擎界就像是新晋的网红&#xff0c;风头一时无两。 搜索引擎的排名算法是确定搜索结果排名的核心&#xff0c;每个搜索引擎都有其独特的排名规则和算法。Bing作为全球第二大搜索引擎&#xff0c;其排名因素和算法同样至关重要。在本…...

使用自定义 PyTorch 运算符优化深度学习数据输入管道

在这篇文章[1]中&#xff0c;我们讨论 PyTorch 对创建自定义运算符的支持&#xff0c;并演示它如何帮助我们解决数据输入管道的性能瓶颈、加速深度学习工作负载并降低训练成本。 构建 PyTorch 扩展 PyTorch 提供了多种创建自定义操作的方法&#xff0c;包括使用自定义模块和/或…...

瑞芯微RKNN开发·yolov5

官方预训练模型转换 下载yolov5-v6.0分支源码解压到本地&#xff0c;并配置基础运行环境。下载官方预训练模型 yolov5n.ptyolov5s.ptyolov5m.pt… 进入yolov5-6.0目录下&#xff0c;新建文件夹weights&#xff0c;并将步骤2中下载的权重文件放进去。修改models/yolo.py文件 …...

Flutter之Widget生命周期

目录 初始化构造函数initStatedidChangeDependencies 运行时builddidUpdateWidget 组件移除deactivatedisposereassemble 函数生命周期说明&#xff1a;实际场景App生命周期 前言&#xff1a;生命周期是一个组件加载到卸载的整个周期&#xff0c;熟悉生命周期可以让我们在合适的…...

机器学习tip:sklearn中的pipeline

文章目录 1 加载数据集2 构思算法的流程3 Pipeline执行流程的分析ReferenceStatement 一个典型的机器学习构建包含若干个过程 源数据ETL数据预处理特征选取模型训练与验证 一个典型的机器学习构建包含若干个过程 以上四个步骤可以抽象为一个包括多个步骤的流水线式工作&…...

Jmeter项目实战

一&#xff0c;性能测试流程 性能需求分析 性能方案设计 业务建模 脚本优化 执行测试 收集性能数据 结果分析 性能测试报告 二&#xff0c;性能需求分析 项目管理系统业务&#xff1a;登录 注册 搜索&#xff08;一般最核心的就是登陆&#xff0c;大多只对登录做压测&a…...

Spring学习笔记注解式开发(3)

Spring学习笔记&#xff08;3&#xff09; 一、Bean的注解式开发1.1、注解开发的基本和Component1.2 注解式开发8.3、Component的三个衍生注解 二、Bean依赖注入注解开发2.1、依赖注入相关注解2.2、Autowired扩展 三、非自定义Bean注解开发四、Bean配置类的注解开发五、Spring注…...

vue3后台管理框架之技术栈

vue3全家桶技术 基础构建&#xff1a; vue3vite4TypeScript 代码格式 &#xff1a; eslintprettystylelint git生命周期钩子&#xff1a; husky css预处理器&#xff1a; sass ui库&#xff1a; element-plus 模拟数据: mock 网络请求&#xff1a; axios 路由&#xff1a; vue…...

7、Linux驱动开发:设备-自动创建设备节点

目录 &#x1f345;点击这里查看所有博文 随着自己工作的进行&#xff0c;接触到的技术栈也越来越多。给我一个很直观的感受就是&#xff0c;某一项技术/经验在刚开始接触的时候都记得很清楚。往往过了几个月都会忘记的差不多了&#xff0c;只有经常会用到的东西才有可能真正记…...

UE5 学习系列(二)用户操作界面及介绍

这篇博客是 UE5 学习系列博客的第二篇&#xff0c;在第一篇的基础上展开这篇内容。博客参考的 B 站视频资料和第一篇的链接如下&#xff1a; 【Note】&#xff1a;如果你已经完成安装等操作&#xff0c;可以只执行第一篇博客中 2. 新建一个空白游戏项目 章节操作&#xff0c;重…...

React Native 导航系统实战(React Navigation)

导航系统实战&#xff08;React Navigation&#xff09; React Navigation 是 React Native 应用中最常用的导航库之一&#xff0c;它提供了多种导航模式&#xff0c;如堆栈导航&#xff08;Stack Navigator&#xff09;、标签导航&#xff08;Tab Navigator&#xff09;和抽屉…...

日语学习-日语知识点小记-构建基础-JLPT-N4阶段(33):にする

日语学习-日语知识点小记-构建基础-JLPT-N4阶段(33):にする 1、前言(1)情况说明(2)工程师的信仰2、知识点(1) にする1,接续:名词+にする2,接续:疑问词+にする3,(A)は(B)にする。(2)復習:(1)复习句子(2)ために & ように(3)そう(4)にする3、…...

《Playwright:微软的自动化测试工具详解》

Playwright 简介:声明内容来自网络&#xff0c;将内容拼接整理出来的文档 Playwright 是微软开发的自动化测试工具&#xff0c;支持 Chrome、Firefox、Safari 等主流浏览器&#xff0c;提供多语言 API&#xff08;Python、JavaScript、Java、.NET&#xff09;。它的特点包括&a…...

vscode(仍待补充)

写于2025 6.9 主包将加入vscode这个更权威的圈子 vscode的基本使用 侧边栏 vscode还能连接ssh&#xff1f; debug时使用的launch文件 1.task.json {"tasks": [{"type": "cppbuild","label": "C/C: gcc.exe 生成活动文件"…...

Go 语言接口详解

Go 语言接口详解 核心概念 接口定义 在 Go 语言中&#xff0c;接口是一种抽象类型&#xff0c;它定义了一组方法的集合&#xff1a; // 定义接口 type Shape interface {Area() float64Perimeter() float64 } 接口实现 Go 接口的实现是隐式的&#xff1a; // 矩形结构体…...

SpringBoot+uniapp 的 Champion 俱乐部微信小程序设计与实现,论文初版实现

摘要 本论文旨在设计并实现基于 SpringBoot 和 uniapp 的 Champion 俱乐部微信小程序&#xff0c;以满足俱乐部线上活动推广、会员管理、社交互动等需求。通过 SpringBoot 搭建后端服务&#xff0c;提供稳定高效的数据处理与业务逻辑支持&#xff1b;利用 uniapp 实现跨平台前…...

【git】把本地更改提交远程新分支feature_g

创建并切换新分支 git checkout -b feature_g 添加并提交更改 git add . git commit -m “实现图片上传功能” 推送到远程 git push -u origin feature_g...

.Net Framework 4/C# 关键字(非常用,持续更新...)

一、is 关键字 is 关键字用于检查对象是否于给定类型兼容,如果兼容将返回 true,如果不兼容则返回 false,在进行类型转换前,可以先使用 is 关键字判断对象是否与指定类型兼容,如果兼容才进行转换,这样的转换是安全的。 例如有:首先创建一个字符串对象,然后将字符串对象隐…...

Fabric V2.5 通用溯源系统——增加图片上传与下载功能

fabric-trace项目在发布一年后,部署量已突破1000次,为支持更多场景,现新增支持图片信息上链,本文对图片上传、下载功能代码进行梳理,包含智能合约、后端、前端部分。 一、智能合约修改 为了增加图片信息上链溯源,需要对底层数据结构进行修改,在此对智能合约中的农产品数…...