基本信息
源码名称:生产者消费者(WindowsPC.cpp)
源码大小:5.57KB
文件格式:.cpp
开发语言:C/C++
更新时间:2020-12-07
   友情提示:(无需注册或充值,赞助后即可获取资源下载链接)

     嘿,亲!知识可是无价之宝呢,但咱这精心整理的资料也耗费了不少心血呀。小小地破费一下,绝对物超所值哦!如有下载和支付问题,请联系我们QQ(微信同号):813200300

本次赞助数额为: 2 元 
   源码介绍


# include <windows.h>
# include <stdio.h>
# include <stdlib.h>
# include <time.h>


PROCESS_INFORMATION StartClone(int nCloneID);

int main(int argc, char* argv[] )
{
	int nClone=0 ;
	int i,time,j;
	HANDLE SEM_FULL;
	HANDLE SEM_EMPTY;
	HANDLE SEM_MUTEX;
	HANDLE hMap;
	PROCESS_INFORMATION nH[5];
	int *pData,*out;
    if (argc > 1)
	{
		// 从第二个参数中提取克隆ID
		sscanf(argv[1] , "%d" , &nClone) ;
	}
		  
	if(nClone==0)     //主进程,创建5个子进程,前2个为生产者,后3个为消费者                  
	{
		//建立信号量
		SEM_FULL=CreateSemaphore(NULL,0,3,"FULL");
		SEM_EMPTY=CreateSemaphore(NULL,3,3,"EMPTY");
		SEM_MUTEX=CreateSemaphore(NULL,1,1,"MUTEX");

		//建立共享内存
		HANDLE CurrentProcess = GetCurrentProcess();

		hMap = CreateFileMapping (
			    CurrentProcess,     //在当前进程中创建文件映射
				NULL,               //默认的安全性
				PAGE_READWRITE,      //可读写权
				0,                   //最大尺寸(高32位)
				sizeof(int)*4,       //最小尺寸(低32位)
				"buffer");           //该文件映射的作为共享内存的缓冲区,取名为“buffer”

		//在文件映射上创建视图
		pData = (int*)MapViewOfFile (
			               hMap,            //保存文件的对象
						   FILE_MAP_WRITE,  //映射可读可写
						   0,               //在文件的开头处(高32位)开始
						   0,               //在文件的开头处(低32位)
						   sizeof(int)*4);  //整个文件要映射4个字节
		
		if(pData != NULL)
		{
			ZeroMemory(pData,sizeof(int)*4);     //分配内存空间,并清零
		}

		out=pData;

		//建立5个子进程
		for (i=0;i<5;i  )
		{
			nH[i]=StartClone(  nClone);
		}

		//等待子进程结束
		for(i=0;i<5;i  )
			WaitForSingleObject(nH[i].hProcess,INFINITE);
		
		//关闭子进程句柄
		for (i=0;i<5;i  )
		{
			CloseHandle(nH[i].hProcess);
			CloseHandle(nH[i].hThread);
		}
		//关闭句柄
		CloseHandle(SEM_MUTEX);
        CloseHandle(hMap);
        CloseHandle(SEM_EMPTY);
       	CloseHandle(SEM_FULL);
	}	
	else if (nClone > 0 &&nClone < 3)            //2个生产者
	{
		for (i=0;i<6;i  )
		{
			//获得句柄
			SEM_EMPTY=OpenSemaphore(SEMAPHORE_ALL_ACCESS,NULL,"EMPTY");
			SEM_FULL=OpenSemaphore(SEMAPHORE_ALL_ACCESS,NULL,"FULL");
			SEM_MUTEX=OpenSemaphore(SEMAPHORE_ALL_ACCESS,NULL,"MUTEX");

			hMap = OpenFileMapping(FILE_MAP_WRITE,FALSE,"buffer");
			pData = (int*)MapViewOfFile (hMap, FILE_MAP_WRITE, 0, 0, sizeof(int)*4);
            out=pData;

			//p 申请信号量
			WaitForSingleObject(SEM_EMPTY, INFINITE);
			WaitForSingleObject(SEM_MUTEX, INFINITE);
  
			//向缓冲区添加产品,把0置为1
			(*pData)  ;
			*(pData (*pData))=1;
			
			//获得时间
			SYSTEMTIME curtime;
			GetSystemTime(&curtime);
			
			//输出状态
			printf("生产者%d 写入数据 时间:%02d:%02d:%02d:%03d.\n "  
			          ,nClone,curtime.wHour,curtime.wMinute,curtime.wSecond,curtime.wMilliseconds);
			printf(" 缓存区内容为: ");
			for (j=1;j<=3;j  )
			{
				printf("%4d",*(out j));
			}
			printf("\n");

			//释放信号量
			ReleaseSemaphore(SEM_MUTEX,1,NULL);
			ReleaseSemaphore(SEM_FULL,1,NULL);	

			//关闭句柄
			CloseHandle(SEM_MUTEX);
			CloseHandle(SEM_EMPTY);
			CloseHandle(SEM_FULL);
            CloseHandle(hMap);

			//随机等待
			
			time = rand()%450;
			Sleep(time);
		}
	}
	else if(nClone>2 && nClone< 6)              //3个消费者
	{
		//获得句柄
		for (i=0;i<4;i  )
		{
			SEM_EMPTY=OpenSemaphore(SEMAPHORE_ALL_ACCESS,NULL,"EMPTY");
			SEM_FULL=OpenSemaphore(SEMAPHORE_ALL_ACCESS,NULL,"FULL");
			SEM_MUTEX=OpenSemaphore(SEMAPHORE_ALL_ACCESS,NULL,"MUTEX");

			hMap = OpenFileMapping(FILE_MAP_WRITE,FALSE,"buffer");
			pData = (int*)MapViewOfFile (hMap, FILE_MAP_WRITE, 0, 0, sizeof(int)*4);
            out=pData;

			WaitForSingleObject(SEM_FULL, INFINITE);
			WaitForSingleObject(SEM_MUTEX, INFINITE);
			

			SYSTEMTIME curtime;
			GetSystemTime(&curtime);

            //取产品,将1置为0
			*(pData (*pData))=0;
			(*pData)--;

			//输出当前信息
			printf("消费者%d 取出数据 时间:%02d:%02d:%02d:%03d.\n"
				     ,nClone-2,curtime.wHour,curtime.wMinute,curtime.wSecond,curtime.wMilliseconds);

			
            printf(" 缓存区内容为: ");
			for (j=1;j<=3;j  )
			{
				printf("%4d",*(out j));
			}
			printf("\n");

			//释放信号量
			ReleaseSemaphore(SEM_MUTEX,1,NULL);
			ReleaseSemaphore(SEM_EMPTY,1,NULL);

			//关闭句柄
			CloseHandle(SEM_MUTEX);
			CloseHandle(SEM_EMPTY);
			CloseHandle(SEM_FULL);
            CloseHandle(hMap);

			//随机等待
			time = rand()%1000;
			Sleep(time);
		}
	}
	return 0;
}    


// 创建一个克隆的进程并赋于其ID值,并返回进程和线程信息
PROCESS_INFORMATION StartClone(int nCloneID)
{
	// 获得用于当前可执行文件的文件名
	TCHAR szFilename[MAX_PATH] ;
	GetModuleFileName(NULL, szFilename, MAX_PATH) ;
	
	//创建子进程命令行的格式化,获得应用程序的EXE文件名并克隆进程的ID值
	TCHAR szCmdLine[MAX_PATH] ;
	sprintf(szCmdLine, "\"%s\" %d", szFilename, nCloneID) ;
	
	STARTUPINFO si;   // 用于子进程的STARTUPINFO结构
	ZeroMemory(reinterpret_cast <void*> (&si) , sizeof(si) ) ;
	// reinterpret_cast为数据类型转换操作
	si.cb = sizeof(si) ;					
	PROCESS_INFORMATION pi;      // 说明一个用于记录子进程的相关信息的结构变量
	
	// 利用同样的可执行文件和命令行创建进程
	BOOL bCreateOK = CreateProcess(
		szFilename,				// 可执行的应用程序的名称
		szCmdLine,				// 指定创建一个子进程的符号标识
		NULL,					// 缺省的进程安全性
		NULL,					// 缺省的线程安全性
		FALSE,					// 不继承打开文件的句柄
		NULL,	             	// 不使用新的控制台
		NULL,					// 新的环境
		NULL,					// 当前目录
		&si,					// 启动信息
		&pi) ;					// 返回进程和线程信息
	// 运行结束,关闭进程和其线程的句柄
	return pi;
}