#include <stdio.h>
#include <process.h>
#include <Windows.h>
//信号量与关键段
CRITICAL_SECTION g_cs;
HANDLE Empty,Full; const int BUFFER_SIZE=;//10个缓冲池
const int numofp=;//生产者线程数
const int numofc=;//消费者线程数
const int numofpr=;//要生产的产品数
int buffer[BUFFER_SIZE];
int pf=,pe=;
int count=;
int count_c=;
int mp=; //生产者
unsigned int __stdcall Producer(PVOID p){ while(true){
//等待有空的缓冲区出现
WaitForSingleObject(Empty, INFINITE);
//互斥的访问缓冲区
EnterCriticalSection(&g_cs);
if(count>numofpr){
LeaveCriticalSection(&g_cs);
ReleaseSemaphore(Full,,NULL);
break;
}
buffer[pe]=++mp;
//printf("%d,",*((int *)p));
printf("编号为%d的生产者从第%d个空缓冲池中写入数据%d\n",*((int *)p),pe,buffer[pe]);
pe=(pe+)%BUFFER_SIZE;
count++;
LeaveCriticalSection(&g_cs);
//通知消费者有新数据了
ReleaseSemaphore(Full,,NULL);
}
return ;
}
//消费者
unsigned int __stdcall Consumer(PVOID p){ while(true){
WaitForSingleObject(Full,INFINITE);
EnterCriticalSection(&g_cs);
if(count_c>numofpr){
LeaveCriticalSection(&g_cs);
ReleaseSemaphore(Empty,,NULL);
break;
}
//printf("%d,",*((int *)p));
printf(" 编号为%d的消费者从第%d个FULL缓冲区中取出数据%d\n",*((int *)p),pf,buffer[pf]);
pf=(pf+)%BUFFER_SIZE;
count_c++;
LeaveCriticalSection(&g_cs);
ReleaseSemaphore(Empty,,NULL);
}
return ;
} int main(){
printf(" 生产者消费者问题:%d生产者-%d消费者-%d缓冲区-%d个产品\n\n",numofp,numofc,BUFFER_SIZE,numofpr);
InitializeCriticalSection(&g_cs);//初始化一个临界区对象
//初始化信号量,一个记录有产品的缓冲区个数,另一个记录空缓冲区个数.
Empty=CreateSemaphore(NULL,,,NULL);
Full=CreateSemaphore(NULL,,,NULL);
const int THREADNUM=numofc+numofp;//线程数
HANDLE hThread[THREADNUM];
//生产者线程
int a=,b=,c=,d=;
hThread[]=(HANDLE)_beginthreadex(
NULL,//安全属性,NULL为默认安全属性
,//指定线程堆栈的大小,一般为0
Producer,//指定线程函数的地址
&a,//传递给线程的参数的指针
,//线程初始状态,0:立即运行
NULL);//用于记录线程ID的地址 hThread[]=(HANDLE)_beginthreadex(NULL,,Producer,&b,,NULL); hThread[]=(HANDLE)_beginthreadex(NULL,,Consumer,&c,,NULL); hThread[]=(HANDLE)_beginthreadex(NULL,,Consumer,&d,,NULL);
WaitForMultipleObjects(THREADNUM, hThread, TRUE, INFINITE);
for (int i = ; i < THREADNUM; i++)
CloseHandle(hThread[i]); //销毁信号量和关键段
CloseHandle(Empty);
CloseHandle(Full);
DeleteCriticalSection(&g_cs);
system("pause"); return ;
}

生产者消费者问题c语言实现的更多相关文章

  1. go语言实现生产者-消费者

    前言: 之前在学习操作系统的时候,就知道生产者-消费者,但是概念是模模糊糊的,好像是一直没搞明白. 其实很简单嘛,生产者生产,消费者进行消费,就是如此简单.了解了一下go语言的goroute,感觉实现 ...

  2. Go语言协程并发---生产者消费者实例

    package main import ( "fmt" "strconv" "time" ) /* 改进生产者消费者模型 ·生产者每秒生产一 ...

  3. LabVIEW之生产者/消费者模式--队列操作 彭会锋

    LabVIEW之生产者/消费者模式--队列操作 彭会锋 本文章主要是对学习LabVIEW之生产者/消费者模式的学习笔记,其中涉及到同步控制技术-队列.事件.状态机.生产者-消费者模式,这几种技术在在本 ...

  4. java 生产者消费者问题 并发问题的解决

    引言 生产者和消费者问题是线程模型中的经典问题:生产者和消费者在同一时间段内共用同一个存储空间,如下图所示,生产者向空间里存放数据,而消费者取用数据,如果不加以协调可能会出现以下情况: 生产者消费者图 ...

  5. 通过生产者消费者模式例子讲解Java基类方法wait、notify、notifyAll

    wait(),notify()和notifyAll()都是Java基类java.lang.Object的方法. 通俗解释wait():在当前线程等待其它线程唤醒.notify(): 唤醒一个线程正在等 ...

  6. 生产者/消费者问题的多种Java实现方式--转

    实质上,很多后台服务程序并发控制的基本原理都可以归纳为生产者/消费者模式,而这是恰恰是在本科操作系统课堂上老师反复讲解,而我们却视而不见不以为然的.在博文<一种面向作业流(工作流)的轻量级可复用 ...

  7. Java实现生产者消费者问题与读者写者问题

    摘要: Java实现生产者消费者问题与读者写者问题 1.生产者消费者问题 生产者消费者问题是研究多线程程序时绕不开的经典问题之一,它描述是有一块缓冲区作为仓库,生产者可以将产品放入仓库,消费者则可以从 ...

  8. java 生产者消费者问题 并发问题的解决(转)

    引言 生产者和消费者问题是线程模型中的经典问题:生产者和消费者在同一时间段内共用同一个存储空间,如下图所示,生产者向空间里存放数据,而消费者取用数据,如果不加以协调可能会出现以下情况: 生产者消费者图 ...

  9. (转)生产者/消费者问题的多种Java实现方式 (待整理)

    实质上,很多后台服务程序并发控制的基本原理都可以归纳为生产者/消费者模式,而这是恰恰是在本科操作系统课堂上老师反复讲解,而我们却视而不见不以为然的.在博文<一种面向作业流(工作流)的轻量级可复用 ...

随机推荐

  1. Oracle中的AS和IS

    Oracle中的AS和IS是ORACLE为了方便而设置的同义词基本上没有不同 . 使用规则: 1.在创建存储过程(PROCEDURE)/函数(FUNCTION),以及自定义类型(TPYE)和包(PAC ...

  2. 【UML】——为什么要使用UML

    以前一提到UML,就想到了复杂的流程图.很敬佩哪些想想就能画出整个系统的UML图的人,因为他们头脑中有整个软件架构的蓝图,这样在编写实现的时候,就会知道哪个地方改怎么做,哪个地方如何扩展. 而想成为架 ...

  3. 剑指OFFER之第一个只出现一次的字符(九度OJ1283)

    题目描述: 在一个字符串(1<=字符串长度<=10000,全部由大写字母组成)中找到第一个只出现一次的字符. 输入: 输入有多组数据每一组输入一个字符串. 输出: 输出第一个只出现一次的字 ...

  4. Mac联网恢复系统重新安装Lion

    Mac的Lion系统,虽然不像Windows那样需要经常重装,但也难免会有要重置的时候,比如更换硬盘.本文介绍如何利用Mac的联网恢复系统进行Lion系统的在线恢复.Mac的在线恢复系统只在近几年的机 ...

  5. Behavioral模式State模式

    1.意向 同意一个目标,然后改变其内部状态,改变它的行为. 对象似乎改变它的类别. 2.别名 状态对象(Objects for States) 3.动机 考虑一个表示网络连接的类TCPConnecti ...

  6. javascript---遇到关于this的相关问题(解决this)(持续更新中...)

    1.在原型中使用this <!doctype html> <html lang="en"> <head> <meta charset=&q ...

  7. 面试题总结之Linux/Shell

    Linux Linux cshrc文件作用 Linux如何起进程/查看进程/杀进程 Linux 文件755 代表什么权限 Linux辅助线程 Linux进程间通信方法 pipeline,msgq... ...

  8. ubuntu 13.04 root权限设置方法详解

    很多朋友安装升级Ubuntu 13.04之后不知道ubuntu 13.04 root权限设置的具体方法,今天这篇文章就将为大家详细介绍设置root权限的步骤,新手朋友可以来看一看哦~ Ubunto 1 ...

  9. SurfaceView的使用

    1.概念 SurfaceView是View类的子类,可以直接从内存或者DMA等硬件接口取得图像数据,是个非常重要的绘图视图.它的特性是:可以在主线程之外的 线程中向屏幕绘图上.这样可以避免画图任务繁重 ...

  10. ppss

    parallel processing shell script Oct 19 Q: how to schedule multi-cpus on each event?