设计模式 之 生产消费者模型 (C++)

文章目录

  • 设计模式 之 生产消费者模型 (C++)
    • 引言
    • 生产消费者模型的基本概念
    • 为什么需要生产消费者模型
    • 应用场景:
    • C++ 实现生产消费者模型
      • 代码示例
      • 代码详细解释
        • 共享资源和同步机制
        • 生产者函数 `producer()`
        • 消费者函数 `consumer()`
        • 主函数 `main()`
    • 注意事项
    • 总结

设计模式 之 生产消费者模型 (C++)

引言

在多线程编程的世界里,生产消费者模型是一个经典且实用的设计模式。它能够高效地解决多个线程之间的数据共享和协作问题,在很多实际场景中都有广泛的应用,比如消息队列系统、数据处理流水线等。本文将深入探讨 C++ 中生产消费者模型的原理、实现方式以及相关的注意事项。

生产消费者模型的基本概念

生产消费者模型主要包含三个核心部分:生产者、消费者和缓冲区。

  • 生产者:负责生成数据或任务,并将其放入缓冲区。
  • 缓冲区:作为生产者和消费者之间的共享区域,用于临时存储生产者产生的数据,起到解耦和协调生产者与消费者速度差异的作用。
  • 消费者:从缓冲区中取出数据或任务进行处理。

为什么需要生产消费者模型

想象一下,如果没有缓冲区,生产者和消费者直接进行交互,那么它们的执行速度必须严格匹配。一旦生产者生产速度过快,消费者可能来不及处理;反之,若消费者处理速度过快,生产者又可能跟不上节奏。这就会导致程序的效率低下,甚至出现数据丢失或线程阻塞等问题。而引入缓冲区后,生产者和消费者可以独立运行,各自按照自己的速度进行生产和消费,大大提高了程序的并发性能和灵活性。

应用场景:

在软件开发里,Web 服务器把客户端请求作为生产者数据存入请求队列,工作线程作为消费者处理请求,提升并发处理能力;图形图像处理软件中,读取图像数据的线程是生产者,处理数据的线程是消费者,实现读写并行。系统设计方面,消息队列系统里生产者服务发消息,消费者服务订阅消费,实现异步解耦;数据库读写分离时,写操作是生产者将请求入队,写线程或服务器作为消费者执行。

C++ 实现生产消费者模型

代码示例

#include <iostream>
#include <queue>
#include <thread>
#include <mutex>
#include <condition_variable>
#include <chrono>// 定义一个共享队列作为缓冲区
std::queue<int> buffer;
// 定义互斥锁,用于保护共享资源
std::mutex mtx;
// 定义条件变量,用于线程间的同步
std::condition_variable not_full;
std::condition_variable not_empty;
// 定义缓冲区的最大容量
const int MAX_SIZE = 5;// 生产者函数
void producer() {for (int i = 0; i < 10; ++i) {std::unique_lock<std::mutex> lock(mtx);// 等待缓冲区有空闲位置not_full.wait(lock, [] { return buffer.size() < MAX_SIZE; });// 生产数据buffer.push(i);std::cout << "生产者生产了数据 " << i << std::endl;// 通知消费者缓冲区有新数据not_empty.notify_one();lock.unlock();// 模拟生产时间std::this_thread::sleep_for(std::chrono::seconds(1));}
}// 消费者函数
void consumer() {while (true) {std::unique_lock<std::mutex> lock(mtx);// 等待缓冲区有数据not_empty.wait(lock, [] { return!buffer.empty(); });// 消费数据int data = buffer.front();buffer.pop();std::cout << "消费者消费了数据 " << data << std::endl;// 通知生产者缓冲区有空闲位置not_full.notify_one();lock.unlock();// 模拟消费时间std::this_thread::sleep_for(std::chrono::seconds(2));}
}int main() {// 创建生产者和消费者线程std::thread producer_thread(producer);std::thread consumer_thread(consumer);// 等待生产者线程结束producer_thread.join();// 由于消费者线程是无限循环,这里简单等待一段时间后强制结束程序std::this_thread::sleep_for(std::chrono::seconds(20));// 可以考虑更优雅的方式结束消费者线程return 0;
}

代码详细解释

共享资源和同步机制
  • std::queue<int> buffer:这是一个标准库中的队列,作为生产者和消费者之间的共享缓冲区。它可以存储整数类型的数据,你可以根据实际需求将其改为存储其他类型的数据。
  • std::mutex mtx:互斥锁是线程同步的重要工具,用于保护对共享资源(这里是 buffer)的访问。在多线程环境中,多个线程可能同时尝试访问和修改 buffer,使用互斥锁可以确保同一时间只有一个线程能够对其进行操作,避免数据竞争和不一致的问题。
  • std::condition_variable not_fullstd::condition_variable not_empty:条件变量用于线程间的同步通信。not_full 用于通知生产者缓冲区有空闲位置,当缓冲区已满时,生产者线程会等待这个条件变量;not_empty 用于通知消费者缓冲区有新数据,当缓冲区为空时,消费者线程会等待这个条件变量。
  • MAX_SIZE:定义了缓冲区的最大容量,避免缓冲区无限增长导致内存溢出。
生产者函数 producer()
  • std::unique_lock<std::mutex> lock(mtx):使用 std::unique_lock 来管理互斥锁。它会在构造时自动锁定互斥锁,在析构时自动解锁,确保锁的正确使用,避免忘记解锁导致死锁。
  • not_full.wait(lock, [] { return buffer.size() < MAX_SIZE; }):这是条件变量的核心用法。wait 函数会先释放互斥锁,然后阻塞当前线程,直到条件变量被通知且谓词(这里是 buffer.size() < MAX_SIZE)为真。当其他线程调用 not_full.notify_one()not_full.notify_all() 时,该线程会被唤醒,重新获取互斥锁并检查谓词。如果谓词为真,则继续执行后续代码;否则,继续等待。
  • buffer.push(i):将生产的数据放入缓冲区。
  • not_empty.notify_one():通知一个等待在 not_empty 条件变量上的消费者线程,缓冲区有新数据可供消费。
  • lock.unlock():手动解锁互斥锁,因为 std::this_thread::sleep_for 期间不需要持有锁,释放锁可以让其他线程有机会访问共享资源。
  • std::this_thread::sleep_for(std::chrono::seconds(1)):模拟生产数据所需的时间。
消费者函数 consumer()
  • 与生产者函数类似,使用 std::unique_lock 锁定互斥锁,调用 not_empty.wait 等待缓冲区有数据。
  • int data = buffer.front(); buffer.pop();:从缓冲区取出数据进行消费。
  • not_full.notify_one():通知一个等待在 not_full 条件变量上的生产者线程,缓冲区有空闲位置。
  • 同样使用 std::this_thread::sleep_for 模拟消费数据所需的时间。
主函数 main()
  • 创建生产者和消费者线程,并启动它们。
  • 使用 producer_thread.join() 等待生产者线程结束。
  • 由于消费者线程是一个无限循环,这里简单地等待 20 秒后程序结束。在实际应用中,需要更优雅的方式来结束消费者线程,例如使用标志位或其他同步机制。

注意事项

  • 线程安全:在多线程编程中,线程安全是至关重要的。确保对共享资源的访问都使用互斥锁进行保护,避免数据竞争和不一致的问题。
  • 条件变量的使用:条件变量的 wait 函数一定要结合谓词使用,避免虚假唤醒。虚假唤醒是指线程在没有收到明确通知的情况下被唤醒,使用谓词可以确保线程只有在满足特定条件时才会继续执行。
  • 线程结束处理:消费者线程通常是一个无限循环,需要考虑如何优雅地结束它。可以使用一个标志位,当生产者线程结束后,设置该标志位,消费者线程在每次循环时检查该标志位,若标志位被设置,则退出循环。

总结

生产消费者模型是一种强大的多线程编程模式,通过引入缓冲区和同步机制,实现了生产者和消费者之间的高效协作。在 C++ 中,我们可以使用标准库提供的互斥锁和条件变量来实现线程安全的生产消费者模型。在实际应用中,要根据具体需求合理设计缓冲区的大小和生产消费的逻辑,同时注意线程安全和线程结束的处理,以确保程序的稳定性和性能。希望本文能帮助你更好地理解和应用生产消费者模型。

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

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

相关文章

Spring Boot 项目开发流程全解析

目录 引言 一、开发环境准备 二、创建项目 三、项目结构 四、开发业务逻辑 1.创建实体类&#xff1a; 2.创建数据访问层&#xff08;DAO&#xff09;&#xff1a; 3.创建服务层&#xff08;Service&#xff09;&#xff1a; 4.创建控制器层&#xff08;Controller&…

数据结构课程设计(java实现)---九宫格游戏,也称幻方

【问题描述】 九宫格&#xff0c;一款数字游戏&#xff0c;起源于河图洛书&#xff0c;与洛书是中国古代流传下来的两幅神秘图案&#xff0c;历来被认为是河洛文化的滥觞&#xff0c;中华文明的源头&#xff0c;被誉为"宇宙魔方"。九宫格游戏对人们的思维锻炼有着极大…

GPT-4.5 怎么样?如何升级使用ChatGPTPlus/Pro? GPT-4.5设计目标是成为一款非推理型模型的巅峰之作

GPT-4.5 怎么样&#xff1f;如何升级使用ChatGPTPlus/Pro? GPT-4.5设计目标是成为一款非推理型模型的巅峰之作 今天我们来说说上午发布的GPT-4.5&#xff0c;接下来我们说说GPT4.5到底如何&#xff0c;有哪些功能&#xff1f;有哪些性能提升&#xff1f;怎么快速使用到GPT-4.…

【vscode-解决方案】vscode 无法登录远程服务器的两种解决办法

解决方案一&#xff1a; 查找原因 命令 ps ajx | grep vscode 可能会看到一下这堆信息&#xff08;如果没有大概率不是这个原因导致&#xff09; 这堆信息的含义&#xff1a;当你使用 vscode 远程登录服务器时&#xff0c;我们远程机器服务端要给你启动一个叫做 vscode serv…

一、对4*3按键模块编程分析

一、4*3键盘模块实物分析 说明&#xff1a; 1、横着4排&#xff0c;竖着3列&#xff0c;加起来共7组&#xff0c;所以对外引出7根线。 2、根据排针终端引脚又可分两类。即横排和竖列对应的引脚。 二、代码编写构想&#xff1a; 1、使用7个gpio输入中断&#xff0c;检测7个…

自然语言处理NLP入门 -- 第十节NLP 实战项目 2: 简单的聊天机器人

一、为什么要做聊天机器人&#xff1f; 在互联网时代&#xff0c;我们日常接触到的“在线客服”“自动问答”等&#xff0c;大多是以聊天机器人的形式出现。它能帮我们快速回复常见问题&#xff0c;让用户获得及时的帮助&#xff0c;并在一定程度上减少人工客服的压力。 同时&…

linux(1)文件管理

文章目录 文件目录系统相对路径绝对路径命令解析器文件管理 文件目录系统 bin&#xff1a; 二进制文件目录&#xff0c;存储可执行文件 dev&#xff1a;设备目录&#xff0c;所有的硬件都会抽象成文件存储&#xff0c;比如鼠标键盘 home&#xff1a;存储普通用户的家目录 li…

CSS—选择器详解:5分钟动手掌握选择器

个人博客&#xff1a;haichenyi.com。感谢关注 1. 目录 1–目录2–引言3–种类4–优先级 引言 什么是选择器&#xff1f; CSS选择器是CSS&#xff08;层叠样式表&#xff09;中的一种规则&#xff0c;用于指定要应用样式的HTML元素。它们就像是指向网页中特定元素的指针&#…

大模型微调入门(Transformers + Pytorch)

目标 输入&#xff1a;你是谁&#xff1f; 输出&#xff1a;我们预训练的名字。 训练 为了性能好下载小参数模型&#xff0c;普通机器都能运行。 下载模型 # 方式1&#xff1a;使用魔搭社区SDK 下载 # down_deepseek.py from modelscope import snapshot_download model_…

DeepSeek实战

DeepSeek 接入实战&#xff1a;从零开始快速上手 引言 在当今的 AI 领域&#xff0c;DeepSeek 作为一个强大的自然语言处理&#xff08;NLP&#xff09;平台&#xff0c;提供了丰富的 API 接口&#xff0c;帮助开发者快速实现智能对话、文本生成、语义分析等功能。本文将带你…

Android NDK打包封装教程与优化技巧

关于NDK打包封装的问题。首先,用户可能不太清楚NDK的基本概念,所以我应该先解释NDK是什么以及它的作用。然后,用户可能想知道如何在Android项目中使用NDK,所以需要分步骤说明配置过程,包括安装NDK、配置CMake或ndk-build,创建JNI接口,编写C/C++代码,编译和打包。 接下…

【告别双日期面板!一招实现el-date-picker智能联动日期选择】

告别双日期面板&#xff01;一招实现el-date-picker智能联动日期选择 1.需求背景2.DateTimePicker 现状图3.日期选择器实现代码4.日期选择器实现效果图5.日期时间选择器实现代码6.日期时间选择器实现效果图 1.需求背景 在用户使用时间查询时&#xff0c;我们经常需要按月份筛选…

Linux(ftrace)__mcount的实现原理

Linux 内核调试工具ftrace 之&#xff08;_mcount的实现原理&#xff09; ftrace 是 Linux 内核中的一种跟踪工具&#xff0c;主要用于性能分析、调试和内核代码的执行跟踪。它通过在内核代码的关键点插入探针&#xff08;probe&#xff09;来记录函数调用和执行信息。这对于开…

Java注解(Annotation)

一、注解的定义 核心概念 注解是Java中一种特殊形式的“元数据”&#xff0c;用于为类、方法、字段、参数等代码元素附加说明信息。它不会直接影响代码逻辑&#xff0c;但可以通过编译器、框架或反射机制进行解析和处理。 与注释&#xff08;Comment&#xff09;的区别 注释&a…

tauri2+typescript+vue+vite+leaflet等的简单联合使用(一)

项目目标 主要的目的是学习tauri。 流程 1、搭建项目 2、简单的在项目使用leaflet 3、打包 准备项目 环境准备 废话不多说&#xff0c;直接开始 需要有准备能运行Rust的环境和Node&#xff0c;对于Rust可以参考下面这位大佬的文章&#xff0c;Node不必细说。 Rust 和…

深入解析 Svelte:下一代前端框架的革命

深入解析 Svelte&#xff1a;下一代前端框架的革命 1. Svelte 简介 Svelte 是一款前端框架&#xff0c;与 React、Vue 等传统框架不同&#xff0c;它采用 编译时&#xff08;Compile-time&#xff09; 方式来优化前端应用。它不像 React 或 Vue 依赖虚拟 DOM&#xff0c;而是…

关于流水线的理解

还是不太理解&#xff0c;我之前一直以为&#xff0c;对axis总线&#xff0c;每一级的寄存器就像fifo一样&#xff0c;一级一级的分级存储最后一级需要的数据。 像这张图&#xff0c;一开始是在解析axis流形式的数据包&#xff0c;数据包一直都能输入&#xff0c;所以valid一直…

Python代码之美:从规范到艺术

基础规范&#xff1a;代码的"颜值"很重要 &#x1f449;大礼包&#x1f381;&#xff1a;&#x1f448; PEP 8&#xff1a;不只是规范&#xff0c;是写作艺术 良好的代码格式就像优美的书法&#xff0c;让人赏心悦目。比如&#xff1a; # 不推荐的写法 def calcul…

【AI+智造】在阿里云Ubuntu 24.04上部署DeepSeek R1 14B的完整方案

作者&#xff1a;Odoo技术开发/资深信息化负责人 日期&#xff1a;2025年2月28日 一、部署背景与目标 DeepSeek R1作为国产大语言模型的代表&#xff0c;凭借其强化学习驱动的推理能力&#xff0c;在复杂任务&#xff08;如数学问题、编程逻辑&#xff09;中表现优异。本地化部…

8 SpringBoot进阶(上):AOP(面向切面编程技术)、AOP案例之统一操作日志

文章目录 前言1. AOP基础1.1 AOP概述: 什么是AOP?1.2 AOP快速入门1.3 Spring AOP核心中的相关术语(面试)2. AOP进阶2.1 通知类型2.1.1 @Around:环绕通知,此注解标注的通知方法在目标方法前、后都被执行(通知的代码在业务方法之前和之后都有)2.1.2 @Before:前置通知,此…