多线程第八篇:生产者消费者问题

2024-08-24 16:18

本文主要是介绍多线程第八篇:生产者消费者问题,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

著名的生产者消费者问题,用到同步和互斥.

  1.我们假设缓冲区大小为2,即最多只能放2个资源,并且肯定大于0.
  2.生产者和消费者是可以同时进入缓冲区的.


代码设计:
1.由于window信号量在缓冲区满的时候无法阻塞,所以需要两个信号量,分别表示缓冲区剩余资源和可用资源分别用于阻塞消费者和生产者.
2.对于全局资源的存取,所有线程都要互斥.
3.对于消费者和生产者来说分别互斥自己的代码.
 

OK上代码:(相当easy)
#include <iostream>
#include <windows.h>
#include <process.h>
int g_count = 0;
CRITICAL_SECTION producer_critical, consumer_critical;
HANDLE producer_semaphore ,consumer_semaphore;
CRITICAL_SECTION all;
unsigned int __stdcall producer(void *)
{
int i =0;
while ( i<5 ){
WaitForSingleObject(consumer_semaphore ,INFINITE);
EnterCriticalSection(&producer_critical );
std::cout <<"生产者生产了."<<std ::endl;
++ i;
EnterCriticalSection(&all );
std::cout <<"生产者"<<++ g_count<<std ::endl;
LeaveCriticalSection(&all );
ReleaseSemaphore(producer_semaphore ,1,NULL);
LeaveCriticalSection(&producer_critical );
}
return 0;
}
unsigned int __stdcall consumer(void *)
{
int i =0;
while ( i<5 ){
WaitForSingleObject(producer_semaphore ,INFINITE);
EnterCriticalSection(&consumer_critical );
std::cout <<"消费者消费了."<<std ::endl;
++ i;
EnterCriticalSection(&all );
std::cout <<"消费者"<<-- g_count<<std ::endl;
LeaveCriticalSection(&all );
ReleaseSemaphore(consumer_semaphore ,1,NULL);
LeaveCriticalSection(&consumer_critical );
}
return 0;
}
int main ()
{
InitializeCriticalSection(& producer_critical);
InitializeCriticalSection(& consumer_critical);
InitializeCriticalSection(& all);
producer_semaphore = CreateSemaphore(NULL ,0,2,NULL);
consumer_semaphore = CreateSemaphore(NULL ,2,2,NULL);
HANDLE hproducer = (HANDLE )_beginthreadex( NULL,0,producer ,NULL,0, NULL);
HANDLE hconsumer = (HANDLE )_beginthreadex( NULL,0,consumer ,NULL,0, NULL);
WaitForSingleObject( hproducer,INFINITE );
WaitForSingleObject( hconsumer,INFINITE );
CloseHandle( hproducer);
CloseHandle( hconsumer);
return 0;
}

 
看出来上述结果说明的问题了吗???
我们来分析一下:
1.生产者和消费者执行的代码并不是互斥的(除了对全局资源的控制:EnterCriticalSection (&all );)
 
2.数字总是大于0,证明消费者总是在有资源的情况下采取的,而小于3,证明消费者在缓冲区满的情况下是不放的.


这篇关于多线程第八篇:生产者消费者问题的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1102975

相关文章

解决pandas无法读取csv文件数据的问题

《解决pandas无法读取csv文件数据的问题》本文讲述作者用Pandas读取CSV文件时因参数设置不当导致数据错位,通过调整delimiter和on_bad_lines参数最终解决问题,并强调正确参... 目录一、前言二、问题复现1. 问题2. 通过 on_bad_lines=‘warn’ 跳过异常数据3

解决RocketMQ的幂等性问题

《解决RocketMQ的幂等性问题》重复消费因调用链路长、消息发送超时或消费者故障导致,通过生产者消息查询、Redis缓存及消费者唯一主键可以确保幂等性,避免重复处理,本文主要介绍了解决RocketM... 目录造成重复消费的原因解决方法生产者端消费者端代码实现造成重复消费的原因当系统的调用链路比较长的时

深度解析Nginx日志分析与499状态码问题解决

《深度解析Nginx日志分析与499状态码问题解决》在Web服务器运维和性能优化过程中,Nginx日志是排查问题的重要依据,本文将围绕Nginx日志分析、499状态码的成因、排查方法及解决方案展开讨论... 目录前言1. Nginx日志基础1.1 Nginx日志存放位置1.2 Nginx日志格式2. 499

kkFileView启动报错:报错2003端口占用的问题及解决

《kkFileView启动报错:报错2003端口占用的问题及解决》kkFileView启动报错因office组件2003端口未关闭,解决:查杀占用端口的进程,终止Java进程,使用shutdown.s... 目录原因解决总结kkFileViewjavascript启动报错启动office组件失败,请检查of

RabbitMQ消费端单线程与多线程案例讲解

《RabbitMQ消费端单线程与多线程案例讲解》文章解析RabbitMQ消费端单线程与多线程处理机制,说明concurrency控制消费者数量,max-concurrency控制最大线程数,prefe... 目录 一、基础概念详细解释:举个例子:✅ 单消费者 + 单线程消费❌ 单消费者 + 多线程消费❌ 多

SpringBoot 异常处理/自定义格式校验的问题实例详解

《SpringBoot异常处理/自定义格式校验的问题实例详解》文章探讨SpringBoot中自定义注解校验问题,区分参数级与类级约束触发的异常类型,建议通过@RestControllerAdvice... 目录1. 问题简要描述2. 异常触发1) 参数级别约束2) 类级别约束3. 异常处理1) 字段级别约束

Python错误AttributeError: 'NoneType' object has no attribute问题的彻底解决方法

《Python错误AttributeError:NoneTypeobjecthasnoattribute问题的彻底解决方法》在Python项目开发和调试过程中,经常会碰到这样一个异常信息... 目录问题背景与概述错误解读:AttributeError: 'NoneType' object has no at

Spring的RedisTemplate的json反序列泛型丢失问题解决

《Spring的RedisTemplate的json反序列泛型丢失问题解决》本文主要介绍了SpringRedisTemplate中使用JSON序列化时泛型信息丢失的问题及其提出三种解决方案,可以根据性... 目录背景解决方案方案一方案二方案三总结背景在使用RedisTemplate操作redis时我们针对

Kotlin Map映射转换问题小结

《KotlinMap映射转换问题小结》文章介绍了Kotlin集合转换的多种方法,包括map(一对一转换)、mapIndexed(带索引)、mapNotNull(过滤null)、mapKeys/map... 目录Kotlin 集合转换:map、mapIndexed、mapNotNull、mapKeys、map

nginx中端口无权限的问题解决

《nginx中端口无权限的问题解决》当Nginx日志报错bind()to80failed(13:Permissiondenied)时,这通常是由于权限不足导致Nginx无法绑定到80端口,下面就来... 目录一、问题原因分析二、解决方案1. 以 root 权限运行 Nginx(不推荐)2. 为 Nginx