c#多线程之生产者-消费者模型

news2025/1/1 12:47:09

在 C# 中实现 生产者-消费者模式,通常需要多个线程来处理数据的生产和消费。我们可以使用 Queue<T> 来作为存储数据的队列,并使用 ThreadMutexMonitor 来确保线程安全。BlockingCollection<T> 是 C# 提供的一个线程安全的集合,可以非常方便地用于实现生产者-消费者模式。

生产者-消费者模式的关键点:

  1. 生产者线程:产生数据并将其放入队列中。
  2. 消费者线程:从队列中取出数据并进行处理。
  3. 线程同步:使用 BlockingCollection<T> 等线程安全的集合来避免竞争条件,同时确保生产者和消费者之间的协调。

示例:使用 BlockingCollection<T>

C# 提供了 BlockingCollection<T> 类,它可以用来在生产者和消费者线程之间提供同步机制。它是一个线程安全的集合,并支持阻塞操作,因此可以自动协调生产者和消费者的行为。

代码示例:
using System;
using System.Collections.Concurrent;
using System.Linq;
using System.Threading;

class Program
{
    // 使用 BlockingCollection 实现线程安全的队列
    static BlockingCollection<int> queue = new BlockingCollection<int>(5); // 队列最大容量为5

    // 生产者线程
    static void Producer()
    {
        int item = 0;
        while (true)
        {
            Thread.Sleep(500); // 模拟生产延迟

            // 生产数据并加入队列
            queue.Add(item);
            Console.WriteLine("生产者生产数据: " + item);
            item++;
        }
    }

    // 消费者线程
    static void Consumer()
    {
        while (true)
        {
            int item = queue.Take(); // 阻塞直到队列中有数据

            Console.WriteLine("消费者消费数据: " + item);
            Thread.Sleep(1000); // 模拟消费延迟
        }
    }

    static void Main()
    {
        // 启动生产者线程
        Thread producerThread = new Thread(Producer);
        producerThread.Start();

        // 启动消费者线程
        Thread consumerThread = new Thread(Consumer);
        consumerThread.Start();

        // 等待线程结束(实际上,生产者和消费者线程会永远运行下去)
        producerThread.Join();
        consumerThread.Join();
    }
}

代码解释:

  1. BlockingCollection<int> queue:一个线程安全的队列,最大容量为 5。BlockingCollection 会在队列满时阻塞生产者线程,在队列为空时阻塞消费者线程。
  2. Producer():模拟生产者线程,每 500 毫秒生成一个数据并放入队列中。如果队列已满,Add 操作会阻塞生产者线程,直到队列有空位。
  3. Consumer():模拟消费者线程,每秒消费一个数据。Take() 会阻塞直到队列中有数据。
  4. Thread.Sleep():用来模拟生产和消费的延迟。

BlockingCollection<T> 的关键方法:

  • Add(T item):将项目添加到集合中。如果集合已满,它将阻塞直到有空余空间。
  • Take():从集合中移除并返回一个项。如果集合为空,它将阻塞直到有可用项。
  • TryAdd(T item):尝试将项目添加到集合中。如果成功则返回 true,否则返回 false,不会阻塞。
  • TryTake(out T item):尝试从集合中移除并返回一个项。如果集合为空,返回 false

扩展:多个生产者和多个消费者

BlockingCollection<T> 支持多个生产者和多个消费者,并且可以通过它来轻松实现复杂的生产者-消费者模型。你只需要启动多个线程来执行生产者和消费者的逻辑即可。

示例:多个生产者和多个消费者
using System;
using System.Collections.Concurrent;
using System.Threading;

class Program
{
    static BlockingCollection<int> queue = new BlockingCollection<int>(5); // 队列最大容量为5

    // 生产者线程
    static void Producer(int id)
    {
        int item = 0;
        while (true)
        {
            Thread.Sleep(500); // 模拟生产延迟

            // 生产数据并加入队列
            queue.Add(item);
            Console.WriteLine($"生产者 {id} 生产数据: {item}");
            item++;
        }
    }

    // 消费者线程
    static void Consumer(int id)
    {
        while (true)
        {
            int item = queue.Take(); // 阻塞直到队列中有数据

            Console.WriteLine($"消费者 {id} 消费数据: {item}");
            Thread.Sleep(1000); // 模拟消费延迟
        }
    }

    static void Main()
    {
        // 启动多个生产者线程
        for (int i = 1; i <= 2; i++)
        {
            int producerId = i;
            new Thread(() => Producer(producerId)).Start();
        }

        // 启动多个消费者线程
        for (int i = 1; i <= 3; i++)
        {
            int consumerId = i;
            new Thread(() => Consumer(consumerId)).Start();
        }

        // 主线程等待
        Console.ReadLine();
    }
}

代码解释:

  1. 多个生产者线程:在 Main() 方法中,启动了 2 个生产者线程。每个线程调用 Producer() 方法,生成不同的数据并将其放入共享队列。
  2. 多个消费者线程:启动了 3 个消费者线程,它们从同一个共享队列中取出数据进行处理。

运行结果:

生产者 1 生产数据: 0
生产者 2 生产数据: 0
消费者 1 消费数据: 0
生产者 1 生产数据: 1
消费者 2 消费数据: 1
消费者 3 消费数据: 2
...

总结:

  • BlockingCollection<T> 是 C# 中实现生产者-消费者模式的理想工具。它是线程安全的,支持阻塞操作,且可以容纳多个生产者和消费者。
  • 通过 BlockingCollection<T>AddTake 方法,生产者和消费者可以安全地进行数据交换而无需担心并发问题。
  • 使用多个生产者和消费者线程时,BlockingCollection<T> 会自动处理队列的同步和线程间协调。

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

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

相关文章

【机器学习(九)】分类和回归任务-多层感知机(Multilayer Perceptron,MLP)算法-Sentosa_DSML社区版 (1)11

文章目录 一、算法概念11二、算法原理&#xff08;一&#xff09;感知机&#xff08;二&#xff09;多层感知机1、隐藏层2、激活函数sigma函数tanh函数ReLU函数 3、反向传播算法 三、算法优缺点&#xff08;一&#xff09;优点&#xff08;二&#xff09;缺点 四、MLP分类任务实…

R基于贝叶斯加法回归树BART、MCMC的DLNM分布滞后非线性模型分析母婴PM2.5暴露与出生体重数据及GAM模型对比、关键窗口识别

全文链接&#xff1a;https://tecdat.cn/?p38667 摘要&#xff1a;在母婴暴露于空气污染对儿童健康影响的研究中&#xff0c;常需对孕期暴露情况与健康结果进行回归分析。分布滞后非线性模型&#xff08;DLNM&#xff09;是一种常用于估计暴露 - 时间 - 响应函数的统计方法&am…

e3 1220lv3 cpu-z分数

e3 1220lv3 双核四线程&#xff0c;1.1G频率&#xff0c;最低可在800MHZ运行&#xff0c;TDP 13W。 使用PE启动后测试cpu-z分数。 现在e3 1220lv3的价格落到69元。

Debian安装配置RocketMQ

安装配置 本次安装在/tools/rocket目录下 下载 wget https://dist.apache.org/repos/dist/release/rocketmq/5.3.1/rocketmq-all-5.3.1-bin-release.zip 解压缩 unzip rocketmq-all-5.3.1-bin-release.zip 如果出现以下报错 -bash: unzip: command not found可安装unzip工具后执…

cocos 运行时,实时查看、修改节点树

简介 cocos论坛提供的一款辅助查看、修改cocos运行时的节点树工具&#xff0c;同时也可以实时修改运行的节点树并进行修改。在此感谢大佬提供这么实用的工具。 参考链接&#xff1a;【运行时显示节点树插件】ccc-devtools悄悄更新 - Creator 2.x - Cocos中文社区 仓库链接&a…

UE(虚幻)学习(四) 第一个C++类来控制小球移动来理解蓝图和脚本如何工作

UE5视频看了不少&#xff0c;但基本都是蓝图如何搞&#xff0c;或者改一下属性&#xff0c;理解UE系统现有组件使用的。一直对C脚本和蓝图之间的关系不是很理解&#xff0c;看到一个视频讲的很好&#xff0c;我也做笔记记录一下。 我的环境是UE5.3.2. 创建UE空项目 我们创建…

记一次内存泄漏分析(待写稿)

背景 线上Flink频繁重启&#xff0c;先后排查了很多情况&#xff0c;目前在内存阶段排查&#xff0c;首先说说学到的知识 内存泄漏分析 JVM常用命令 JConsole JVisualvm 快照的这里是最有用的&#xff0c;它和jmap不同&#xff0c;jmap查找的是占用字节最多的类&#xff…

【蓝桥杯——物联网设计与开发】拓展模块5 - 光敏/热释电模块

目录 一、光敏/热释电模块 &#xff08;1&#xff09;资源介绍 &#x1f505;原理图 &#x1f505;AS312 &#x1f319;简介 &#x1f319;特性 &#x1f505;LDR &#xff08;2&#xff09;STM32CubeMX 软件配置 &#xff08;3&#xff09;代码编写 &#xff08;4&#x…

C语言从入门到放弃教程

C语言从入门到放弃 1. 介绍1.1 特点1.2 历史与发展1.3 应用领域 2. 安装2.1 编译器安装2.2 编辑器安装 3. 第一个程序1. 包含头文件2. 主函数定义3. 打印语句4. 返回值 4. 基础语法4.1 注释4.1.1 单行注释4.1.2 多行注释 4.2 关键字4.2.1 C语言标准4.2.2 C89/C90关键字&#xf…

第三百四十六节 JavaFX教程 - JavaFX绑定

JavaFX教程 - JavaFX绑定 JavaFX绑定同步两个值&#xff1a;当因变量更改时&#xff0c;其他变量更改。 要将属性绑定到另一个属性&#xff0c;请调用bind()方法&#xff0c;该方法在一个方向绑定值。例如&#xff0c;当属性A绑定到属性B时&#xff0c;属性B的更改将更新属性A…

慧集通客户案例:致远OA与熵基考勤机集成方案

本原型公司是一家专注大健康产业的综合性高新科技形实体企业&#xff0c;按照单位的战略业务布局&#xff0c;围绕“做强做优、世界一流”的目标&#xff0c;加快内外部资源整合、加强业务协同、优化资源配置&#xff0c;有序推进大健康及相关产业的有机融合&#xff0c;加快构…

SCSA: Exploring the Synergistic Effects Between Spatial and Channel Attention

摘要 https://arxiv.org/pdf/2407.05128 通道注意力和空间注意力分别为各种下游视觉任务在提取特征依赖性和空间结构关系方面带来了显著改进。通道注意力和空间注意力的结合使用被广泛认为有利于进一步提升性能&#xff1b;然而&#xff0c;通道注意力和空间注意力之间的协同作…

UE5在蓝图中使用VarestX插件访问API

在Fab中安装好VarestX免费插件 这个插件可以用来远程请求http和api等&#xff0c;返回json等格式内容 插件网址 https://www.fab.com/zh-cn/listings/d283e40c-4ee5-4e73-8110-cc7253cbeaab 虚幻里开启插件 然后网上随便搜个免费api测试一下&#xff0c;这里我找了个微博热搜…

碰一碰发视频矩阵系统源码搭建,支持OEM

一、引言 随着短视频的火爆发展&#xff0c;碰一碰发视频的矩阵系统逐渐受到关注。这种系统能够实现用户通过碰一碰设备&#xff08;如 NFC 标签&#xff09;快速触发视频的发布&#xff0c;在营销推广、互动体验等领域有着广泛的应用前景。本文将详细介绍碰一碰发视频矩阵系统…

Pandas01

文章目录 内容简介1 常用数据分析三方库2 Jupyter notebook3 Series的创建3.1 通过Numpy的Ndarray 创建一个Series3.2 通过列表创建Series 4 Series的属性和方法4.1 常用属性4.2 常用方法4.3 布尔值列表筛选部分数据4.4 Series 的运算 5 DataFrame的创建通过字典创建通过列表[元…

WebPack3项目升级webpack5的配置调试记录

文章目录 前言一、webpack3环境1.1、知识点记录1.1.1、配置解释1.1.2、webpack与sass版本对应关系1.1.3、CommonJS与ESModule1.1.4、node版本管理nvm1.1.5、sass-loader、sass与node-sass 1.2、其他1.2.1、.d.ts是什么文件1.2.2、react与types/react版本对应关系1.2.3、webpack…

plsql :用户system通过sysdba连接数据库--报错ora-01031

一、winR cmd通过命令窗口登录sys用户 sql sys/[password]//localhost:1521/[service_name] as sysdba二、输入用户名:sys as sysdba 三、输入密码:自己设的 四、执行grant sysdba to system; 再去PL/SQL连接就可以了

Quo Vadis, Anomaly Detection? LLMs and VLMs in the Spotlight 论文阅读

文章信息&#xff1a; 原文链接&#xff1a;https://arxiv.org/abs/2412.18298 Abstract 视频异常检测&#xff08;VAD&#xff09;通过整合大语言模型&#xff08;LLMs&#xff09;和视觉语言模型&#xff08;VLMs&#xff09;取得了显著进展&#xff0c;解决了动态开放世界…

药片缺陷检测数据集,8625张图片,使用YOLO,PASICAL VOC XML,COCO JSON格式标注,可识别药品是否有缺陷,是否完整

药片缺陷检测数据集&#xff0c;8625张图片&#xff0c;使用YOLO&#xff0c;PASICAL VOC XML&#xff0c;COCO JSON格式标注&#xff0c;可识别药品是否有缺陷&#xff0c;是否完整 有缺陷的标注信息&#xff1a; 无缺陷的标注信息 数据集下载&#xff1a; yolov11:https://d…

蓝桥杯速成教程{三}(adc,i2c,uart)

目录 一、adc 原理图​编辑引脚配置 Adc通道使能配置 实例测试 ​编辑效果显示 案例程序 badc 按键相关函数 测量频率占空比 main 按键的过程 显示界面的过程 二、IIC通信-eeprom 原理图AT24C02 引脚配置 不可用状态&#xff0c;用的软件IIC 官方库移植 At24c02手册 ​编辑…