实验二生产者和消费者_第1页
实验二生产者和消费者_第2页
实验二生产者和消费者_第3页
实验二生产者和消费者_第4页
实验二生产者和消费者_第5页
已阅读5页,还剩3页未读 继续免费阅读

下载本文档

版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领

文档简介

1.1实验目标掌握操作系统对进程管理的同步和互斥问题,以经典同步问题“生产者和消费者”为例,运用所学C语言知识,编写该问题的模拟运行环境。1.2实验环境教学机房,C语言程序平台。1.3实验内容及步骤1.了解经典同步问题“生产者和消费者”生产者与消费者可以通过一个环形缓冲池联系起来,环形缓冲池由几个大小相等的缓冲块组成,每个缓冲块容纳一个产品。每个生产者可不断地每次往缓冲池中送一个生产产品,而每个消费者则可不断地每次从缓冲池中取出一个产品。指针i和指针j分别指出当前的第一个空缓冲块和第一个满缓冲块。2.分析和理解(1)既存在合作同步问题,也存在临界区互斥问题合作同步:当缓冲池全满时,表示供过于求,生产者必须等待,同时唤醒消费者;当缓冲池全空时,表示供不应求,消费者应等待,同时唤醒生产者。互斥:缓冲池显然是临界资源,所在生产者与消费都要使用它,而且都要改变它的状态。(2)基于环形缓冲区的生产者与消费者关系形式描述:公用信号量mutex:初值为1,用于实现临界区互斥生产者私用信号量empty:初值为n,指示空缓冲块数目消费者私用信号量full:初值为0,指示满缓冲块数目整型量i和j初值为0,i指示首空缓冲块序号,j指示首满缓冲块序号(3)PV原语varmutex,empty,full:semaphore;i,j:integer;buffer:array[0...n-1]ofitem;i:=j:=1;Procedureproducer;beginwhiletruedobeginproduceaproduct;P(empty);P(mutex);buffer(i):=product;i:=(i+1)modn;V(mutex);V(full);end;end;Procedureconsumer;beginP(full);P(mutex);goods:=buffer(j);j:=(j+1)modn;V(mutex);V(empty);consumeaproduct;end;end;3.用C语言编程搭建“生产者和消费者”经典进程通信问题的环境。要求程序运行时,按任意键停止,显示当前系统的各个参数的值。提交实验报告,以及相关程序列表。打包成附件上传。#include<windows.h>#include<stdio.h>#include<stdlib.h>#defineP(S)WaitForSingleObject(S,INFINITE)//定义Windows下的P操作#defineV(S)ReleaseSemaphore(S,1,NULL)//定义Windows下的V操作#definerate1000#defineCONSUMER_NUM10/*消费者个数*/#definePRODUCER_NUM10/*生产者个数*/#defineBUFFER_NUM4/*缓冲区个数*/typedefHANDLESemaphore;//信号量的Windows原型char*thing[10]={"物品1","物品2","物品3","物品4","物品5","物品6","物品7","物品8","物品9","物品10"};structBuffer{intproduct[BUFFER_NUM];//缓冲区intstart,end;//两个指针}g_buf;Semaphoreg_semBuffer,g_semProduct,g_mutex;//消费者线程DWORDWINAPIConsumer(LPVOIDpara){inti=*(int*)para;//i表示第i个消费者intptr,j;//待消费的内容的指针Sleep(100);while(1){P(g_semProduct);//有产品,先锁住缓冲区P(g_mutex);//记录消费的物品ptr=g_buf.start;//再移动缓冲区指针g_buf.start=(g_buf.start+1)%BUFFER_NUM;V(g_mutex);//让其他消费者或生产者使用g_bufprintf("消费者%d:消费了buf[%d]里的=%s\n",i,ptr,thing[g_duct[ptr]]);Sleep(rate*rand()%10+110);//消费完毕,并释放一个缓冲V(g_semBuffer);if(j++>30)break;}getchar();return0;}//生产者线程DWORDWINAPIProducer(LPVOIDpara){inti=*(int*)para-CONSUMER_NUM;intptr;intdata;//产品Sleep(100);while(1){Sleep(rate*rand()%10+110);data=rand()%10;//等待存放空间P(g_semBuffer);//有地方,先锁住缓冲区P(g_mutex);//记录消费的物品ptr=g_buf.end;//再移动缓冲区指针g_buf.end=(g_buf.end+1)%BUFFER_NUM;//让其他消费者或生产者使用g_bufV(g_mutex);printf("生产者%d:在buf[%d]里放入了%s\n",i,ptr,thing[data]);g_duct[ptr]=data;Sleep(rate/2*rand()%10+110);//放好了完毕,释放一个产品V(g_semProduct);}return0;}intmain(intargc,char*argv[]){//线程技术,前面为消费者线程,后面为生产者线程HANDLEhThread[CONSUMER_NUM+PRODUCER_NUM];//线程计数//srand(time());DWORDtid;inti=0;//初始化信号量g_mutex=CreateSemaphore(NULL,BUFFER_NUM,BUFFER_NUM,"mutexOfConsumerAndProducer");g_semBuffer=CreateSemaphore(NULL,BUFFER_NUM,BUFFER_NUM,"BufferSemaphone");g_semProduct=CreateSemaphore(NULL,0,BUFFER_NUM,"ProductSemaphone");if(!g_semBuffer||!g_semProduct||!g_mutex){printf("CreateSemaphoneError!\n");return-1;}inttotalThreads=CONSUMER_NUM+PRODUCER_NUM;//开启消费者线程for(i=0;i<CONSUMER_NUM;i++){hThread[i]=CreateThread(NULL,0,Consumer,&i,0,&tid);if(hThread[i])WaitForSingleObject(hThread[i],10);}//开启生产者线程for(;i<totalThreads;i++){hThread[i]=CreateThread(NULL,0,Producer,&i,0,&tid);if(hThread[i])WaitForSingleObject(hThread[i],10);}//生产者和消费者的执行WaitForMultipleObjects(totalThreads,hThread,TRUE,INFINITE);return0;}1.4实验思考题1.思考在“生产者和消费者”经典同步问题中,两个P操作是否可以互换位置,以及两个V操作是否可以互换位置。在生产者-消费者问题中,如果将两个P操作,即P(full)和P(mutex)互换位置,或者P(empty)和P(mutex)互换位置,都可能引起死锁。考虑系统中缓冲区全满时,若以生产者进程先执行了P(mutex)操作并获得成功,当再执行P(empty)操作时,他将因失败而进入阻塞状态,它期待消费者执行V(empty)来唤醒自己。在此之前,它不可能执行V(mutex)操作,从而使企图通过P(mutex)进入自己的临界区的其他生产者和所有消费者进程全部进入阻塞状态,从而引起系统死锁。类似地,消费者进程若先执行P(mutex),后执行P(full),同样可能造成死锁。V(full)和V(mutex)互换位置,或者V(empty)和V(mutex)互换位置,则不会引起死锁,其影响只是使临界资源的释放略微推迟一些。2.思考在“哲学家就餐”经典同步问题中,如何修改程序,可以保证不会发生死锁现象。(1)至多只允许有四位哲学家同时去拿左边的筷子,最终能保证至少有一位哲学家能够进餐,并在用毕是能释放出他用过的两只筷子,从而使更多的哲学家能够进餐。(2)仅当哲学家的左、右两只筷子均可使用时,才允许他拿起筷子进餐。(3)规定奇数号哲学家先拿他左边的筷子,然后再去拿右边的筷子,而偶数号哲学家则相反。按此规定,将是1、2号哲学家竞争1号筷子;3、4号哲学家竞争3号筷子。即五位哲学家都先竞争奇数号筷子,获得后,再去竞争偶数号筷子,最后总会有一位哲学家能获得两只筷子而进餐。3.思考在“读者与写者”经典同步问题中,如何修改程序,变为“写者优先”的算法。(写者优先算法)Varrmutex,wmutex,mutex,s:semaphore=1,1,1,1;Writecount:integer:=0;Reader:beginRepeatWait(s);Wait(rmutex);Ifreadcount=0thenwait(wmutex);Readcount:readcount+1;Signal(rmutex);Signal(s);Performreadoperation;Wait(rmutex);Readcount:=readcount:=readcount-1;Ifreadcount=0thensignal(wmutex);Signal(rmutex);Untilfalse;EndWriter:beginRepeatWait(mutex);Ifwritecount=0thenwait(s);Writecount:writecount+1;Signal(mutex);Wait(wmutex);Performwriteoperatin;Signal(wmutex);Wait(mutex);Writecount=0thensignal(s);Signal(mutex);Untilfalse;end4.分析以下进程运行

温馨提示

  • 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
  • 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
  • 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
  • 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
  • 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
  • 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
  • 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。

评论

0/150

提交评论