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

【Go】通道:缓冲通道和非缓冲通道

目录

通道的基本概念

缓冲通道

非缓冲通道

总结


通道的基本概念

在Go语言中,通道是一种特殊的类型,用于在goroutine之间传递数据。你可以将通道想象为数据的传输管道。通道分为两种类型:

  1. 非缓冲通道(Unbuffered Channels):发送操作会阻塞,直到另一goroutine在对应的通道上执行接收操作,这时候数据才会被发送成功,发送goroutine才能继续执行。
  2. 缓冲通道(Buffered Channels):可以存储一定数量的值,无需立即有goroutine接收这些值。

缓冲通道

        缓冲通道通过在make函数中指定第二个参数来创建,这个参数定义了通道可以存储的值的数量。在你提供的代码中,messages是一个缓冲通道,其容量为2:

messages := make(chan string, 2)

        这意味着即使没有goroutine准备好接收数据,你也可以往messages通道发送两个字符串。如果尝试发送更多的数据,那么发送操作会阻塞,直到有空间可用。

messages <- "buffered"
messages <- "channel"fmt.Println(<-messages)
fmt.Println(<-messages)

        简单代码:


package mainimport "fmt"func main() {// Here we `make` a channel of strings buffering up to// 2 values.messages := make(chan string, 2)// Because this channel is buffered, we can send these// values into the channel without a corresponding// concurrent receive.messages <- "buffered"messages <- "channel"// Later we can receive these two values as usual.fmt.Println(<-messages)fmt.Println(<-messages)
}

使用场景

        缓冲通道在需要解耦发送者和接收者的速率时非常有用。例如,在一个情况下,可能生产者生成数据的速度快于消费者处理数据的速度,那么一个适当大小的缓冲可以帮助平衡这种速率差异,减少直接阻塞的发生。

        这里将展示一个使用缓冲通道的Go程序,该程序模拟一个简单的并行任务处理场景,其中多个工人(goroutine)并发地从一个任务队列(缓冲通道)中获取任务并执行。此示例中,我们将创建一个任务通道和一个结果通道,每个工人都从任务通道接收任务,处理完毕后将结果发送到结果通道。这种方式非常适用于需要任务分发和结果收集的场景。

package mainimport ("fmt""time""sync"
)// Task 表示一个简单的任务,这里仅仅是一个数值
type Task int// Result 表示任务处理的结果,这里包含任务原始值和处理后的信息
type Result struct {Task TaskInfo string
}func worker(id int, tasks <-chan Task, results chan<- Result, wg *sync.WaitGroup) {defer wg.Done()for task := range tasks {fmt.Printf("工人 %d 开始处理任务 %d\n", id, task)// 模拟任务处理时间time.Sleep(time.Second)// 发送结果到结果通道results <- Result{Task: task, Info: fmt.Sprintf("工人 %d 完成", id)}}
}func main() {// 创建缓冲通道tasks := make(chan Task, 10)results := make(chan Result, 10)// 使用 WaitGroup 来等待所有工人完成var wg sync.WaitGroup// 启动三个工人 goroutinesfor i := 1; i <= 3; i++ {wg.Add(1)go worker(i, tasks, results, &wg)}// 分发任务for j := 1; j <= 5; j++ {tasks <- Task(j)}// 关闭任务通道,表示不再有任务被发送close(tasks)// 等待所有工人完成wg.Wait()// 关闭结果通道close(results)// 输出所有处理结果for result := range results {fmt.Printf("任务 %d: %s\n", result.Task, result.Info)}
}

程序说明

  1. 任务和结果结构:我们定义了两种类型,TaskResult,分别代表任务和处理结果。
  2. 工人goroutineworker 函数是每个工人的行为定义。它持续从任务通道接收任务,处理它们(这里仅模拟为等待1秒),然后将结果发送到结果通道。
  3. 主函数中的并发执行
    • 创建并初始化缓冲通道 tasksresults
    • 启动3个工人goroutine,每个都在独立的线程中执行。
    • 发送5个任务到任务通道。
    • 关闭任务通道,告知工人不再有新任务。
    • 使用sync.WaitGroup来等待所有工人的任务处理完成。
    • 关闭结果通道并打印所有结果。

        这个例子展示了如何使用缓冲通道来管理并发任务的分配和结果收集,使得多个工人能够并行处理任务,同时主程序能够等待所有任务完成并最终收集所有的处理结果。这种模式在实际开发中非常有用,尤其是在需要并行处理大量数据或任务的应用场景中。继续努力,不断深化对Go并发编程的理解和应用!

非缓冲通道

        Go语言中的非缓冲通道是一种在goroutine之间进行通信的机制,它在发送和接收数据时具有同步的特性。在非缓冲通道上,数据的发送必须有对应的接收操作同时准备好,否则发送操作会阻塞,直到有goroutine来接收数据。同样,如果通道中没有数据,接收操作也会阻塞,直到有数据被发送到通道。这种特性使得非缓冲通道不仅是数据传输的渠道,还是一个同步多个goroutine的强大工具。

基础介绍

        非缓冲通道保证了数据传递的即时性,即数据发送后立即被接收。这种即时的数据交换机制意味着每个发送操作都必须有一个对应的接收操作准备接收数据,这种机制在并发编程中常用于控制不同goroutine之间的执行顺序。

        下面的示例展示了如何使用非缓冲通道进行两个goroutine之间的同步通信。在这个例子中,我们将创建一个主goroutine和一个工作goroutine,主goroutine发送一个任务到工作goroutine,然后等待工作goroutine的处理结果。

package mainimport ("fmt""time"
)func worker(done chan bool) {fmt.Println("工作中...")time.Sleep(time.Second)fmt.Println("工作完成")// 发送一个值表示工作已经完成done <- true
}func main() {// 创建一个非缓冲的布尔型通道done := make(chan bool)// 启动一个工作goroutinego worker(done)// 等待工作goroutine的通知<-donefmt.Println("在主goroutine中继续执行")
}

程序解析

  1. 通道的创建和使用:我们通过make(chan bool)创建了一个非缓冲的布尔型通道done。这个通道用来从工作goroutine向主goroutine发送任务完成的信号。
  2. 工作goroutineworker函数模拟长时间运行的任务,完成后通过done通道发送一个true值来通知主goroutine任务已完成。
  3. 主goroutine的阻塞等待:在主goroutine中,使用<-done来阻塞主goroutine的执行,直到从done通道接收到工作完成的信号。

总结

        非缓冲通道是Go语言中一种实现goroutine间同步通信的强大机制。通过确保每个发送操作都必须有一个对应的接收操作同时准备好,非缓冲通道可以精确控制数据的即时传递和goroutine的执行顺序。这种通道不仅是数据传输的渠道,也是协调并发操作的关键工具。通过非缓冲通道,Go程序能够以直接且同步的方式处理并发任务,从而保持高效和可靠的执行流程。简而言之,非缓冲通道是Go并发编程中不可或缺的同步神器

        缓冲通道在Go语言中是一种允许在没有接收方准备好时进行数据传输的通信机制。这种通道通过内部缓冲区来暂存数据,从而允许发送操作在缓冲区未满时立即返回,而不必等待接收方。缓冲通道的存在极大地提升了并发程序的灵活性和效率,使得goroutine之间可以更加灵活地进行非阻塞通信和数据交换。

相关文章:

【Go】通道:缓冲通道和非缓冲通道

目录 通道的基本概念 缓冲通道 非缓冲通道 总结 通道的基本概念 在Go语言中&#xff0c;通道是一种特殊的类型&#xff0c;用于在goroutine之间传递数据。你可以将通道想象为数据的传输管道。通道分为两种类型&#xff1a; 非缓冲通道&#xff08;Unbuffered Channels&…...

Java中数组的使用

在Java编程中&#xff0c;数组是一种非常重要的数据结构&#xff0c;它允许我们存储相同类型的多个元素。对于初学者来说&#xff0c;理解数组的基本概念、初始化、遍历、默认值以及内存分配和使用注意事项是非常关键的。 一、数组的概念 数组是一个可以容纳多个相同类型数据…...

CAP5_Monday

A Set to Max (Easy Version) 给定数组 a 和 b&#xff0c;可以执行以下操作任意次 : 让 a l ∼ a r a_l\sim a_r al​∼ar​ 中的所有所有元素变成 a i a_i ai​ ( l ≤ i ≤ r ) (l\leq i\leq r) (l≤i≤r)&#xff0c; 其中 1 ≤ l ≤ r ≤ n 1\leq l \leq r \leq n 1≤…...

科大讯飞星火开源大模型iFlytekSpark-13B GPU版部署方法

星火大模型的主页&#xff1a;iFlytekSpark-13B: 讯飞星火开源-13B&#xff08;iFlytekSpark-13B&#xff09;拥有130亿参数&#xff0c;新一代认知大模型&#xff0c;一经发布&#xff0c;众多科研院所和高校便期待科大讯飞能够开源。 为了让大家使用的更加方便&#xff0c;科…...

SpringBoot基于RabbitMQ实现消息延迟队列方案

知识小科普 在此之前&#xff0c;简单说明下基于RabbitMQ实现延时队列的相关知识及说明下延时队列的使用场景。 延时队列使用场景 在很多的业务场景中&#xff0c;延时队列可以实现很多功能&#xff0c;此类业务中&#xff0c;一般上是非实时的&#xff0c;需要延迟处理的&a…...

Go语言使用标准库时常见错误

Go的标准库是一组增加和拓展语言的核心包。然而,很容易误用标准库,或者我们对其行为理解有限,导致产生了bug或不应该在生产级应用程序中某些功能。 1. 提供错误的持续时间 标准库提供了获取 time.Duration 的常用函数和方法,但由于 time.Duration 是 int64 的自定义类型,…...

UE5不打包启用像素流 ubuntu22.04

首先查找引擎中像素流的位置&#xff1a; zkzk-ubuntu2023:/media/zk/Data/Linux_Unreal_Engine_5.3.2$ sudo find ./ -name get_ps_servers.sh [sudo] zk 的密码&#xff1a; ./Engine/Plugins/Media/PixelStreaming/Resources/WebServers/get_ps_servers.sh然后在指定路径中…...

Redis 常用数据类型常用命令和应用场景

首先先混个眼熟 Redis 中的 8 种常用数据类型&#xff1a; 5 种基础数据类型&#xff1a;String&#xff08;字符串&#xff09;、List&#xff08;列表&#xff09;、Set&#xff08;集合&#xff09;、Hash&#xff08;散列&#xff09;、Zset&#xff08;有序集合&#xff0…...

ins视频批量下载,instagram批量爬取视频信息

简介 Instagram 是目前最热门的社交媒体平台之一,拥有大量优质的视频内容。但是要逐一下载这些视频往往非常耗时。在这篇文章中,我们将介绍如何使用 Python 编写一个脚本,来实现 Instagram 视频的批量下载和信息爬取。 我们使用selenium获取目标用户的 HTML 源代码,并将其保存…...

Canvas图形编辑器-数据结构与History(undo/redo)

Canvas图形编辑器-数据结构与History(undo/redo) 这是作为 社区老给我推Canvas&#xff0c;于是我也学习Canvas做了个简历编辑器 的后续内容&#xff0c;主要是介绍了对数据结构的设计以及History能力的实现。 在线编辑: https://windrunnermax.github.io/CanvasEditor开源地…...

阿里云Centos7下编译glibc

编译glibc 原来glibc版本 编译前需要的环境: CentOS7 gcc 8.3.0 gdb 8.3.0 make 4.0 binutils 2.39 (ld -v) python 3.6.8 其他看INSTALL, 但有些版本也不易太高 wget https://mirrors.aliyun.com/gnu/glibc/glibc-2.37.tar.gz tar -zxf glibc-2.37.tar.gz cd glibc-2.37/ …...

UE5数字孪生系列笔记(四)

场景的切换 创建一个按钮的用户界面UMG 创建一个Actor&#xff0c;然后将此按钮UMG添加到组件Actor中 调节几个全屏的背景 运行结果 目标点切换功能制作 设置角色到这个按钮的位置效果 按钮被点击就进行跳转 多个地点的切换与旋转 将之前的目标点切换逻辑替换成旋转的逻…...

品牌故事化:Kompas.ai如何塑造深刻的品牌形象

在这个信息爆炸的时代&#xff0c;品牌故事化已经成为企业塑造独特形象、与消费者建立情感联系的重要手段。一个引人入胜的品牌故事不仅能够吸引消费者的注意力&#xff0c;还能够在消费者心中留下持久的印象&#xff0c;建立起强烈的情感连接。本文将深入探讨品牌故事化对于构…...

5g和2.4g频段有什么区别

运行的频段不同 2.4G和5G频段的主要区别在于它们运行的频段不同&#xff0c;2.4G频段运行在2.4GHz的频段上&#xff0c;而5G频段&#xff08;这里指的是5GHz频段&#xff09;运行在5GHz的频段上。12 这导致了两者在传输速度、覆盖范围、抗干扰能力等方面的明显差异。以下是详…...

交通管理在线服务系统|基于Springboot的交通管理系统设计与实现(源码+数据库+文档)

交通管理在线服务系统目录 目录 基于Springboot的交通管理系统设计与实现 一、前言 二、系统功能设计 三、系统实现 1、用户信息管理 2、驾驶证业务管理 3、机动车业务管理 4、机动车业务类型管理 四、数据库设计 1、实体ER图 五、核心代码 六、论文参考 七、最新计…...

konva.js 工具类

konva.js 工具类 class KonvaCanvas {/*** 初始化画布* param {String} domId 容器dom id*/constructor(domId) {this.layer null;this.stage null;this.scale 1;this.init(domId);}/*** 聚焦到指定元素* param {String} elementId 元素dom id*/focusOn(elementId) {if (!t…...

php未能在vscode识别?

在设置里搜php&#xff0c;找到settings.json&#xff0c;设置你的安装路径即可。 成功...

解读MongoDB官方文档获取mongo7.0版本的安装步骤与基本使用

mongo式一款NOSQL数据库&#xff0c;用于存储非结构化数据&#xff0c;mongo是一种用于存储json的数据数据&#xff0c;可以通过mongo提供的命令解析json获取想要的值。 数据模型 了解关系数据库会很熟悉database,table,row,column的概念&#xff0c;分别是数据库&#xff0c…...

【数据结构|C语言版】顺序表

前言1. 初步认识数据结构2. 线性表3. 顺序表3.1 顺序表的概念3.1 顺序表的分类3.2 动态顺序表的实现 结语 前言 各位小伙伴大家好&#xff01;小编来给大家讲解一下数据结构中顺序表的相关知识。 1. 初步认识数据结构 【概念】数据结构是计算机存储、组织数据的⽅式。 数据…...

Unity类银河恶魔城学习记录12-17 p139 In game UI源代码

Alex教程每一P的教程原代码加上我自己的理解初步理解写的注释&#xff0c;可供学习Alex教程的人参考 此代码仅为较上一P有所改变的代码 【Unity教程】从0编程制作类银河恶魔城游戏_哔哩哔哩_bilibili UI.cs using UnityEngine;public class UI : MonoBehaviour {[SerializeFie…...

Android Wi-Fi 连接失败日志分析

1. Android wifi 关键日志总结 (1) Wi-Fi 断开 (CTRL-EVENT-DISCONNECTED reason3) 日志相关部分&#xff1a; 06-05 10:48:40.987 943 943 I wpa_supplicant: wlan0: CTRL-EVENT-DISCONNECTED bssid44:9b:c1:57:a8:90 reason3 locally_generated1解析&#xff1a; CTR…...

调用支付宝接口响应40004 SYSTEM_ERROR问题排查

在对接支付宝API的时候&#xff0c;遇到了一些问题&#xff0c;记录一下排查过程。 Body:{"datadigital_fincloud_generalsaas_face_certify_initialize_response":{"msg":"Business Failed","code":"40004","sub_msg…...

智慧医疗能源事业线深度画像分析(上)

引言 医疗行业作为现代社会的关键基础设施,其能源消耗与环境影响正日益受到关注。随着全球"双碳"目标的推进和可持续发展理念的深入,智慧医疗能源事业线应运而生,致力于通过创新技术与管理方案,重构医疗领域的能源使用模式。这一事业线融合了能源管理、可持续发…...

Neo4j 集群管理:原理、技术与最佳实践深度解析

Neo4j 的集群技术是其企业级高可用性、可扩展性和容错能力的核心。通过深入分析官方文档,本文将系统阐述其集群管理的核心原理、关键技术、实用技巧和行业最佳实践。 Neo4j 的 Causal Clustering 架构提供了一个强大而灵活的基石,用于构建高可用、可扩展且一致的图数据库服务…...

Spring AI与Spring Modulith核心技术解析

Spring AI核心架构解析 Spring AI&#xff08;https://spring.io/projects/spring-ai&#xff09;作为Spring生态中的AI集成框架&#xff0c;其核心设计理念是通过模块化架构降低AI应用的开发复杂度。与Python生态中的LangChain/LlamaIndex等工具类似&#xff0c;但特别为多语…...

DeepSeek 技术赋能无人农场协同作业:用 AI 重构农田管理 “神经网”

目录 一、引言二、DeepSeek 技术大揭秘2.1 核心架构解析2.2 关键技术剖析 三、智能农业无人农场协同作业现状3.1 发展现状概述3.2 协同作业模式介绍 四、DeepSeek 的 “农场奇妙游”4.1 数据处理与分析4.2 作物生长监测与预测4.3 病虫害防治4.4 农机协同作业调度 五、实际案例大…...

代理篇12|深入理解 Vite中的Proxy接口代理配置

在前端开发中,常常会遇到 跨域请求接口 的情况。为了解决这个问题,Vite 和 Webpack 都提供了 proxy 代理功能,用于将本地开发请求转发到后端服务器。 什么是代理(proxy)? 代理是在开发过程中,前端项目通过开发服务器,将指定的请求“转发”到真实的后端服务器,从而绕…...

Aspose.PDF 限制绕过方案:Java 字节码技术实战分享(仅供学习)

Aspose.PDF 限制绕过方案&#xff1a;Java 字节码技术实战分享&#xff08;仅供学习&#xff09; 一、Aspose.PDF 简介二、说明&#xff08;⚠️仅供学习与研究使用&#xff09;三、技术流程总览四、准备工作1. 下载 Jar 包2. Maven 项目依赖配置 五、字节码修改实现代码&#…...

wpf在image控件上快速显示内存图像

wpf在image控件上快速显示内存图像https://www.cnblogs.com/haodafeng/p/10431387.html 如果你在寻找能够快速在image控件刷新大图像&#xff08;比如分辨率3000*3000的图像&#xff09;的办法&#xff0c;尤其是想把内存中的裸数据&#xff08;只有图像的数据&#xff0c;不包…...

如何应对敏捷转型中的团队阻力

应对敏捷转型中的团队阻力需要明确沟通敏捷转型目的、提升团队参与感、提供充分的培训与支持、逐步推进敏捷实践、建立清晰的奖励和反馈机制。其中&#xff0c;明确沟通敏捷转型目的尤为关键&#xff0c;团队成员只有清晰理解转型背后的原因和利益&#xff0c;才能降低对变化的…...