Golang 并发 生产者消费者模式

Golang 并发 生产者消费者模式

生产者-消费者模式能够带来的好处

生产者消费者模式是一种常见的并发编程模式,用于解决生产者和消费者之间的数据传递和处理问题。在该模式中,生产者负责生成数据(生产),而消费者负责处理数据(消费)。生产者和消费者在时间上是解耦的,它们可以独立地以不同的速度执行。生产者消费者模式在并发编程中具有重要性,有以下几个方面的作用:

  • 解耦生产者和消费者: 生产者和消费者之间通过中间的数据缓冲区(如通道)进行通信,从而实现了解耦。生产者和消费者可以独立地进行工作,无需关心对方的状态或执行速度。
  • 平衡资源利用和处理能力: 生产者消费者模式可以平衡生产者和消费者之间的资源利用和处理能力。生产者可以根据消费者的处理能力进行生产,并且消费者可以根据生产者的速度进行消费,从而避免资源的浪费或瓶颈。
  • 提高系统的并发性和响应性: 生产者消费者模式允许并发执行生产者和消费者的任务,从而提高系统的并发性和响应性。通过并发处理数据,可以更好地利用多核处理器和异步执行,从而加快系统的处理速度。
  • 实现异步通信和处理: 生产者消费者模式使得生产者和消费者可以异步地进行数据通信和处理。生产者可以在需要时生成数据,并将其放入缓冲区中,而消费者可以在需要时从缓冲区中获取数据进行处理,从而实现异步的数据交换和处理。
  • 提供可扩展性和模块化: 生产者消费者模式提供了一种可扩展和模块化的设计方式。通过将生产者和消费者解耦,可以方便地添加更多的生产者或消费者,以适应系统需求的变化,同时保持代码的可读性和维护性。
    总之,生产者消费者模式在并发编程中起着重要的作用,通过解耦、平衡资源利用、提高并发性和响应性等方面的优势,可以帮助构建高效、可扩展的并发系统。

应用场景

  • 日志处理: 在日志处理中,可以将日志的生成视为生产者,而日志的消费(如写入文件、发送到远程服务器等)视为消费者。通过使用一个日志通道,生产者可以将日志消息发送到通道,而消费者则从通道中接收日志消息并进行相应的处理。这样可以有效地解耦日志的生成和消费,避免日志处理对业务逻辑的影响。
  • 任务队列: 在某些任务调度和处理场景中,可以使用生产者消费者模式来实现任务队列。生产者负责将任务添加到队列中,而消费者则从队列中获取任务并进行处理。这种方式可以实现任务的异步处理和负载均衡,提高系统的并发性能。
  • 缓存更新: 在某些缓存系统中,生产者消费者模式可用于实现缓存更新的异步处理。当数据发生变化时,生产者负责生成更新请求,而消费者则负责将更新应用到缓存中。通过将更新请求发送到缓存通道,可以实现异步的缓存更新,提高系统的响应性能和吞吐量。

channel的实现

package chapter03import ("fmt""testing""time"
)func producer(ch chan<- int) {for i := 1; i <= 5; i++ {// 将数据发送到通道ch <- ifmt.Println("生产者生产:", i)// 模拟生产过程time.Sleep(time.Second)}close(ch) // 关闭通道
}
func consumer(ch <-chan int, done chan<- bool) {for num := range ch {fmt.Println("消费者消费:", num)// 模拟消费过程time.Sleep(2 * time.Second)}// 通知主线程消费者已完成done <- true
}
func TestRun(t *testing.T) {// 创建带缓冲的通道ch := make(chan int, 3)// 用于通知主线程消费者已完成done := make(chan bool)// 启动生产者goroutinego producer(ch)// 启动消费者goroutinego consumer(ch, done)// 主线程等待消费者完成<-donefmt.Println("消费者已完成")// 主线程结束,程序退出
}

代码输出如下

=== RUN   TestRun
生产者生产: 1
消费者消费: 1
生产者生产: 2
消费者消费: 2
生产者生产: 3
生产者生产: 4
生产者生产: 5
消费者消费: 3
消费者消费: 4
消费者消费: 5
消费者已完成
--- PASS: TestRun (10.00s)
  • producer函数是生产者函数,它通过通道将数据发送到消费者。
  • consumer函数是消费者函数,它从通道中接收数据并进行消费。
  • 主函数是程序的入口,它创建了一个整型通道和一个用于通知消费者完成的通道。

互斥锁和条件变量的实现

package chapter03import ("fmt""sync""testing""time"
)type Data struct {Value int
}
type Queue struct {mutex      sync.Mutexcond       *sync.Condbuffer     []Dataterminated bool
}func NewQueue() *Queue {q := &Queue{}q.cond = sync.NewCond(&q.mutex)return q
}
func (q *Queue) Produce(data Data) {q.mutex.Lock()defer q.mutex.Unlock()q.buffer = append(q.buffer, data)fmt.Printf("Produced: %d\n", data.Value)// 唤醒等待的消费者q.cond.Signal()
}
func (q *Queue) Consume() Data {q.mutex.Lock()defer q.mutex.Unlock()// 等待数据可用for len(q.buffer) == 0 && !q.terminated {q.cond.Wait()}if len(q.buffer) > 0 {data := q.buffer[0]q.buffer = q.buffer[1:]fmt.Printf("Consumed: %d\n", data.Value)return data}return Data{}
}
func (q *Queue) Terminate() {q.mutex.Lock()defer q.mutex.Unlock()q.terminated = true// 唤醒所有等待的消费者q.cond.Broadcast()
}
func TestRun02(t *testing.T) {queue := NewQueue()// 启动生产者for i := 1; i <= 3; i++ {go func(id int) {for j := 1; j <= 5; j++ {data := Data{Value: id*10 + j}queue.Produce(data)time.Sleep(time.Millisecond * 500) // 模拟生产时间}}(i)}// 启动消费者for i := 1; i <= 2; i++ {go func(id int) {for {data := queue.Consume()if data.Value == 0 {break}// 处理消费的数据time.Sleep(time.Millisecond * 1000) // 模拟处理时间}}(i)}// 等待一定时间后终止消费者time.Sleep(time.Second * 6)queue.Terminate()// 等待生产者和消费者完成time.Sleep(time.Second * 1)
}

代码输出如下

=== RUN   TestRun02
Produced: 11
Consumed: 11
Produced: 31
Consumed: 31
Produced: 21
Produced: 32
Produced: 22
Produced: 12
Consumed: 21
Consumed: 32
Produced: 23
Produced: 13
Produced: 33
Produced: 24
Produced: 34
Produced: 14
Consumed: 22
Consumed: 12
Produced: 35
Produced: 15
Produced: 25
Consumed: 23
Consumed: 13
Consumed: 33
Consumed: 24
Consumed: 34
Consumed: 14
Consumed: 35
Consumed: 15
--- PASS: TestRun02 (7.00s)

在上述示例中,

我们创建了一个 Queue 结构体,其中包含了一个互斥锁和一个条件变量。生产者通过 Produce 方法向队列中添加数据,并使用条件变量的 Signal 方法唤醒等待的消费者。消费者通过 Consume 方法从队列中取出数据,如果队列为空且未终止,则通过条件变量的 Wait 方法来阻塞自己。当有数据被生产或终止信号发出时,生产者唤醒等待的消费者。

在主函数中,我们启动了多个生产者和消费者的 goroutine,它们并发地进行生产和消费操作。通过适当的延时模拟生产和消费的时间,展示了生产者和消费者之间的协调工作。

最后,我们通过调用 queue.Terminate() 方法来终止消费者的执行,并通过适当的延时等待生产者和消费者完成。

通过使用互斥锁和条件变量,我们可以实现生产者消费者模式的线程安全同步,确保生产者和消费者之间的正确交互。这种实现方式具有较低的复杂性,并提供了对共享资源的有效管理和控制。

参考

  • https://www.jb51.net/jiaoben/2850675mz.htm

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

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

相关文章

【射影几何13 】梅氏定理和塞瓦定理探讨

梅氏定理和塞瓦定理 目录 一、说明二、梅涅劳斯&#xff08;Menelaus&#xff09;定理三、塞瓦(Giovanni Ceva&#xff09;定理四、塞瓦点的推广4.1 共线定理4.2 三角形外的塞瓦点 一、说明 在射影几何中&#xff0c;梅涅劳斯&#xff08;Menelaus&#xff09;定理和塞瓦定理是…

最大子数组和[中等]

一、题目 给定一个长度为n的环形整数数组nums&#xff0c;返回nums的非空 子数组 的最大可能和 。 环形数组 意味着数组的末端将会与开头相连呈环状。形式上&#xff0c;nums[i]的下一个元素是nums[(i 1) % n]&#xff0c;nums[i]的前一个元素是nums[(i - 1 n) % n]。 子数…

论文封面下划线总是对不齐,这3步你肯定没做!

论文封面 在写论文时&#xff0c;总会遇到论文封面下划线对不齐&#xff0c;学会下面这三招轻松搞定封面。 解决方法 ①选中文字&#xff0c;点击“插入”&#xff0c;选择“表格”&#xff0c;找到“文本转化为表格”。列数为2&#xff0c;文字分割位置选空格&#xff0c;设置…

第21讲:动态内存管理

1.为什么要有动态内存分配 2.malloc和free 3.calloc 4.realloc 5.笔试题 6.总结c/c中程序内存区域划分 1.为什么要有动态内存分配 为了调整申请的空间大小&#xff0c;使程序员可以申请和释放空间&#xff0c;提高程序的灵活性 2.malloc和free 作用&#xff1a;分配一块…

python将.db数据库文件转成Excel文档

python实现.db数据库转Excel 程序实现 上一篇文章程序实现以下功能&#xff1a; 1.读取一个Excel文件,文件名通过函数传参数传入 2.将文件读取的内容保存到一个数据库文件中 3.数据库的文件名以传入的Excel文件的文件名命名 4.将excel文件的工作簿的名字作为数据库的表单名 5…

apache_exporter安装说明

Apache Exporter 问题描述 需要监控apache服务&#xff0c;部署了apache_exporter&#xff0c;对过程进行一下记录。 源码参见apache_exporter ①下载 https://github.com/Lusitaniae/apache_exporter/releases②解压缩 tar -xzvf apache_exporter-0.7.0.linux-amd64.tar…

深入了解Redis:选择适用于你的场景的持久化方案

自然语言处理的发展 文章目录 自然语言处理的发展强烈推荐前言&#xff1a;Redis提供了几种主要的持久化方案&#xff1a;RDB快照持久化&#xff1a;工作原理&#xff1a; AOF日志文件持久化&#xff1a;混合持久化&#xff1a; 总结强烈推荐专栏集锦写在最后 强烈推荐 前些天…

相机图像质量研究(7)常见问题总结:光学结构对成像的影响--镜片固化

系列文章目录 相机图像质量研究(1)Camera成像流程介绍 相机图像质量研究(2)ISP专用平台调优介绍 相机图像质量研究(3)图像质量测试介绍 相机图像质量研究(4)常见问题总结&#xff1a;光学结构对成像的影响--焦距 相机图像质量研究(5)常见问题总结&#xff1a;光学结构对成…

戴上HAUWEI WATCH GT 4,解锁龙年新玩法

春节将至&#xff0c;华为WATCH GT 4作为一款颜值和实力并存的手表&#xff0c;能为节日增添了不少趣味和便利。无论你是钟情于龙年表盘或定制属于自己的表盘&#xff0c;还是过年用来抢红包或远程操控手机拍全家福等等&#xff0c;它都能成为你的“玩伴”。接下来&#xff0c;…

C语言 服务器编程-日志系统

日志系统的实现 引言最简单的日志类 demo按天日志分类和超行日志分类日志信息分级同步和异步两种写入方式 引言 日志系统是通过文件来记录项目的 调试信息&#xff0c;运行状态&#xff0c;访问记录&#xff0c;产生的警告和错误的一个系统&#xff0c;是项目中非常重要的一部…

JUnit 5 注解总结与解析

前言 大家好&#xff0c;我是chowley&#xff0c;通过前篇的JUnit实践&#xff0c;我对这个框架产生了好奇&#xff0c;除了断言判断&#xff0c;它还有哪些用处呢&#xff1f;下面来总结一下它的常见注解及作用。 正文 在Java单元测试中&#xff0c;JUnit是一种常用的测试框…

linux互斥锁:递归锁,非递归锁用法详解

在实际的项目中经常涉及到共享资源&#xff0c;共享资源被多个线程访问会出现竞争现象&#xff1b;为了解决竞争和保护共享资源常用的机制之一就是互斥锁&#xff01; 互斥锁又分为递归锁和非递归锁&#xff0c;互斥锁默认是非递归锁&#xff0c;也是我们常用的上锁方式。那么什…

零基础学Python之整合MySQL

Python 标准数据库接口为 Python DB-API&#xff0c;Python DB-API为开发人员提供了数据库应用编程接口。 不同的数据库你需要下载不同的DB API模块&#xff0c;例如你需要访问Oracle数据库和Mysql数据&#xff0c;你需要下载Oracle和MySQL数据库模块。 DB-API 是一个规范. 它…

【51单片机】要实现动静态数码管,你首先需要知道这些【数码管项目前置知识】

前言 大家好吖&#xff0c;欢迎来到 YY 滴单片机系列 &#xff0c;热烈欢迎&#xff01; 本章主要内容面向接触过单片机的老铁 主要内容含&#xff1a; 本章节内容为【实现动静态数码管】项目的第一个模块完整章节&#xff1a;传送门 欢迎订阅 YY滴C专栏&#xff01;更多干货持…

QT styleSheet——控件设置样式表

QT开发中&#xff0c;需要设置多种多样的控件表现形式&#xff0c;QT实现的styleSheet能够满足多种多样的场景&#xff0c;这里简单的记录下一些我常用的 设置透明背景&#xff0c;鼠标悬浮时&#xff0c;设置背景色&#xff1a; pushButton->setStyleSheet("QPushBu…

Unity接入GVoice腾讯实时语音

Unity接入GVoice腾讯实时语音 一、介绍二、注册GVoice创建项目语音服务1.创建项目2.申请语音权限3.项目管理查看SDK初始化的一些参数和基本信息4.GVoice检测 三、SDK下载SDK是分为两种类型&#xff1a;独立版集成板 SDK放入Unity工程中 四、语音代码写法五、GVoice踩坑语音权限…

【canvas】获取鼠标点击位置坐标的颜色信息

在项目当中&#xff0c;要实现某业务需求例如PS魔棒功能时&#xff0c;则需要获取点击坐标的颜色信息。 功能不复杂&#xff0c;代码也很少&#xff0c;一看便知~~ 核心API为getImageData&#xff0c;传入4个参数&#xff0c;前2个为点击坐标xy&#xff0c;后2个都传1&#xf…

python调用golang中函数方法

一、原因说明&#xff1a;由于simhash方法有多种实现方式&#xff0c;现python中simhash方法与golang中的不一样&#xff0c;需要两者代码生成结果保持一致&#xff0c;故采用python中的代码调用golang编译的so文件来实现。 环境配置&#xff1a;①Windows10系统要有gcc环境&a…

ADSelfService Plus发布离线MFA功能,强化远程工作安全性

ManageEngine ADSelfService Plus推出离线多因素身份验证&#xff0c;提升远程工作安全性确保通过先进的验证方法对企业数据进行授权访问&#xff0c;无论时间、地点或连接问题如何允许远程用户安全进行身份验证&#xff0c;即使未连接到认证服务器或互联网使用高度安全的基于T…

每周编辑精选|希尔贝壳语音数据集系列上线、中国学者建立乳腺癌预后评分系统 MIRS

&#x1f3ee;新春佳节&#xff0c;归程千里为团圆 &#x1f3ee;人间烟火&#xff0c;年味浓浓阖家欢 辞别玉兔&#xff0c;金龙迎春。明晚就是大年三十除夕夜了&#xff01;HyperAI超神经在这里提前祝大家新春快乐&#xff5e;龙行龘龘(d)&#xff01;本周 hyper.ai 官网上线…