生产者与消费者之苹果橘子问题

 生产者与消费者问题

【问题桌上有一空盘,允许存放一只水果。爸爸可向盘中放苹果,也可向盘中放桔子,儿子专等吃盘中的桔子,女儿专等吃盘中的苹果。规定当盘空时一次只能放一只水果供吃者取用,请用PV原语实现爸爸、儿子、女儿三个并发进程的同步。

分析 在本题中,爸爸、儿子、女儿共用一个盘子,盘中一次只能放一个水果。当盘子为空时,爸爸可将一个水果放入果盘中。若放入果盘中的是桔子,则允许儿子吃,女儿必须等待;若放入果盘中的是苹果,则允许女儿吃,儿子必须等待。

本题实际上是生产者-消费者问题的一种变形。这里,生产者放入缓冲区的产品有两类,消费者也有两类,每类消费者只消费其中固定的一类产品。

    :在本题中,应设置三个信号量SSoSa,信号量S表示盘子是否为空,其初值为l;信号量So表示盘中是否有桔子,其初值为0;信号量Sa表示盘中是否有苹果,其初值为0。同步描述如下:

int S1;

int Sa0;

int So0;

      main()

      {

        cobegin

            father();      /*父亲进程*/

            son();        /*儿子进程*/

            daughter();    /*女儿进程*/

        coend

    

    father()

    {

        while(1)

          {

            P(S);

            将水果放入盘中;

            if(放入的是桔子)V(So);

            else  V(Sa);

           }

     }

    son()

    {

        while(1)

          {

             P(So);

             从盘中取出桔子;

             V(S);

             吃桔子;

            

    }

    daughter()

    {

         while(1)

            {

              P(Sa);

              从盘中取出苹果;

              V(S);

              吃苹果;

            


这个问题可以用多线程的方法来解决,即创建父亲(生产者)、儿子(消费者)、女儿(消费者)三个线程。

利用c语言的线程函数

创建线程函数原型:

uintptr_t _beginthread( // NATIVE CODE  
   void( __cdecl *start_address )( void * ),  
   unsigned stack_size,  
   void *arglist   
);  

参数

start_address
启动开始执行新线程的例程的地址。 对于 _beginthread,调用约定是 __cdecl (针对本机代码)或 __clrcall (针对托管代码);对于 _beginthreadex,它是 __stdcall (针对本机代码)或 __clrcall (针对托管代码)。

stack_size
新线程的堆栈大小或 0。

arglist
要传递到新线程的参数列表或 NULL。

/*
*生产者与消费者问题
*苹果橘子问题
*2017.6.11
*/

#include<stdio.h>
#include<process.h>
#include<Windows.h>
#include<time.h>

//定义三个信号量,S代表盘子是否为空,初值为1;So代表盘中是否有橘子,Sa代表盘中是否有苹果,初值为0
int S = 1,So = 0,Sa = 0;
void father(void *p)
{//父亲线程,往盘中放苹果或橘子
	int i = 0;
	while (true)
	{
		Sleep(1000);				//生产延时
		srand(time(NULL));
		i = rand() % 2 + 1;

		if (S == 1)							//盘为空
		{
			S = 0;											//P(S)
			if (i == 1)		//放的是橘子,放水果的随机性怎么解决
			{
				So = 1;										//V(So)
				printf("父亲: 放入了一个橘子\n");
			}
			else {
				Sa = 1;			//放的是苹果				//V(Sa)
				printf("父亲: 放入了一个苹果\n");
			}
		}
	}
	
}

void son(void *p)
{//儿子线程,消费苹果
	while (true)
	{
		if (!S) //如果盘中有水果
		{
			if (Sa == 1)	//是苹果
			{
				Sa = 0;		//消费之
				printf("儿子: 从盘中拿走苹果\n");
				S = 1;		//置盘为空
			}
			Sleep(1000);			//消费延时
		}
	}
}

void daughter(void *p)
{//女儿线程,消费橘子
	while (true)
	{
		if (!S) //如果盘中有水果
		{
			if (So == 1)	//是橘子
			{
				So = 0;		//消费之
				printf("女儿: 从盘中拿走一个橘子\n");
				S = 1;		//盘为空
			}
			Sleep(1000);
		}

	}
}

int main()
{
	int i = 0;
	_beginthread(father, 0, NULL);			//父亲线程
	_beginthread(son, 0, NULL);		//儿子线程
	_beginthread(daughter, 0, NULL);		//女儿线程
	while (true)
	{						//主线程
		scanf_s("%d", &i);
	}

	return 0;
}


运行结果:



  • 13
    点赞
  • 78
    收藏
    觉得还不错? 一键收藏
  • 3
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值