实验2并发与调度2.2Windows2000线程同步(实验估计时间:120分钟)背景知识实验目的工具/准备工作实验内容与步骤背景知识Windows2000提供的常用对象可分成三类:核心应用服务、线程同步和线程间通讯。其中,开发人员可以使用线程同步对象来协调线程和进程的工作,以使其共享信息并执行任务。此类对象包括互锁数据、临界段、事件、互斥体和信号等。多线程编程中关键的一步是保护所有的共享资源,工具主要有互锁函数、临界段和互斥体等;另一个实质性部分是协调线程使其完成应用程序的任务,为此,可利用内核中的事件对象和信号。在进程内或进程间实现线程同步的最方便的方法是使用事件对象,这一组内核对象允许一个线程对其受信状态进行直接控制(见表4-1)。而互斥体则是另一个可命名且安全的内核对象,其主要目的是引导对共享资源的访问。拥有单一访问资源的线程创建互斥体,所有想要访问该资源的线程应该在实际执行操作之前获得互斥体,而在访问结束时立即释放互斥体,以允许下一个等待线程获得互斥体,然后接着进行下去。与事件对象类似,互斥体容易创建、打开、使用并清除。利用CreateMutex()API可创建互斥体,创建时还可以指定一个初始的拥有权标志,通过使用这个标志,只有当线程完成了资源的所有的初始化工作时,才允许创建线程释放互斥体。表4-1用于管理事件对象的APIAPI名称描述CreateEvent()在内核中创建一个新的事件对象。此函数允许有安全性设置、手工还是自动重置的标志以及初始时已接受还是未接受信号状态的标志OpenEvent()创建对已经存在的事件对象的引用。此API函数需要名称、继承标志和所需的访问级别SetEvent()将手工重置事件转化为已接受信号状态ResetEvent()将手工重置事件转化为非接受信号状态PulseEvent()将自动重置事件对象转化为已接受信号状态。当系统释放所有的等待它的线程时此种转化立即发生为了获得互斥体,首先,想要访问调用的线程可使用OpenMutex()API来获得指向对象的句柄;然后,线程将这个句柄提供给一个等待函数。当内核将互斥体对象发送给等待线程时,就表明该线程获得了互斥体的拥有权。当线程获得拥有权时,线程控制了对共享资源的访问——必须设法尽快地放弃互斥体。放弃共享资源时需要在该对象上调用ReleaseMute()API。然后系统负责将互斥体拥有权传递给下一个等待着的线程(由到达时间决定顺序)。实验目的在本实验中,通过对事件和互斥体对象的了解,来加深对Windows2000线程同步的理解。1)回顾系统进程、线程的有关概念,加深对Windows2000线程的理解。2)了解事件和互斥体对象。3)通过分析实验程序,了解管理事件对象的API。4)了解在进程中如何使用事件对象。5)了解在进程中如何使用互斥体对象。6)了解父进程创建子进程的程序设计方法。工具/准备工作在开始本实验之前,请回顾教科书的相关内容。您需要做以下准备:1)一台运行Windows2000Professional操作系统的计算机。2)计算机中需安装VisualC++6.0专业版或企业版。实验内容与步骤1.事件对象2.互斥体对象1.事件对象清单2-1程序展示了如何在进程间使用事件。父进程启动时,利用CreateEvent()API创建一个命名的、可共享的事件和子进程,然后等待子进程向事件发出信号并终止父进程。在创建时,子进程通过OpenEvent()API打开事件对象,调用SetEvent()API使其转化为已接受信号状态。两个进程在发出信号之后几乎立即终止。步骤1:登录进入Windows2000Professional。步骤2:在“开始”菜单中单击“程序”-“MicrosoftVisualStudio6.0”–“MicrosoftVisualC++6.0”命令,进入VisualC++窗口。步骤3:在工具栏单击“打开”按钮,在“打开”对话框中找到并打开实验源程序2-1.cpp。清单2-1创建和打开事件对象在进程间传送信号//event项目#includewindows.h#includeiostream//以下是句柄事件。实际中很可能使用共享的包含文件来进行通讯staticLPCTSTRg_szContinueEvent=w2kdg.EventDemo.event.Continue;//本方法只是创建了一个进程的副本,以子进程模式(由命令行指定)工作BOOLCreateChild(){//提取当前可执行文件的文件名TCHARszFilename[MAX_PATH];::GetModuleFileName(NULL,szFilename,MAX_PATH);//格式化用于子进程的命令行,指明它是一个EXE文件和子进程TCHARszCmdLine[MAX_PATH];::sprintf(szCmdLine,\%s\child,szFilename);//子进程的启动信息结构STARTUPINFOsi;::ZeroMemory(reinterpret_castvoid*(&si),sizeof(si));si.cb=sizeof(si);//必须是本结构的大小//返回的子进程的进程信息结构PROCESS_INFORMATIONpi;//使用同一可执行文件和告诉它是一个子进程的命令行创建进程BOOLbCreateOK=::CreateProcess(szFilename,//生成的可执行文件名szCmdLine,//指示其行为与子进程一样的标志NULL,//子进程句柄的安全性NULL,//子线程句柄的安全性FALSE,//不继承句柄0,//特殊的创建标志NULL,//新环境NULL,//当前目录&si,//启动信息结构&pi);//返回的进程信息结构//释放对子进程的引用if(bCreateOK){::CloseHandle(pi.hProcess);::CloseHandle(pi.hThread);}return(bCreateOK);}//下面的方法创建一个事件和一个子进程,然后等待子进程在返回前向事件发出信号voidWaitForChild(){//createaneweventobjectforthechildprocess//tousewhenreleasingthisprocessHANDLEhEventContinue=::CreateEvent(NULL,//缺省的安全性,子进程将具有访问权限TRUE,//手工重置事件FALSE,//初始时是非接受信号状态g_szContinueEvent);//事件名称if(hEventContinue!=NULL){std::couteventcreatedstd::endl;//创建子进程if(::CreateChild()){std::coutchlidcreatedstd::endl;//等待,直到子进程发出信号std::coutParentwaitingonchild.std::endl;::WaitForSingleObject(hEventContinue,INFINITE);::Sleep(1500);//删去这句试试std::coutparentreceivedtheenventsignalingfromchildstd::endl;}//清除句柄::CloseHandle(hEventContinue);hEventContinue=INVALID_HANDLE_VALUE;}}//以下方法在子进程模式下被调用,其功能只是向父进程发出终止信号voidSignalParent(){//尝试打开句柄std::coutchildprocessbegining......std::endl;HANDLEhEventContinue=::OpenEvent(EVENT_MODIFY_STATE,//所要求的最小访问权限FALSE,//不是可继承的句柄g_szContinueEvent);//事件名称if(hEventContinue!=NULL){::SetEvent(hEventContinue);std::couteventsignaledstd::endl;}//清除句柄::CloseHandle(hEventContinue);hEventContinue=INVALID_HANDLE_VALUE;}intmain(intargc,char*argv[]){//检查父进程或是子进程是否启动if(argc1&&::strcmp(argv[1],child)==0){//向父进程创建的事件发出信号::SignalParent();}else{//创建一个事件并等待子进程发出信号::WaitForChild();::Sleep(1500);std::coutParentreleased.std::endl;}return0;}步骤4:单击“Build”菜单中的“Compile2-1.cpp”命令,并单击“是”按钮确认。系统对2-1.cpp进行编译。步骤5:编译完成后,单击“Build”菜单中的“Build2-1.exe”命令,建立2-1.exe可执行文件。操作能否正常进行?如果不行,则可能的原因是什么?操作能正常进行步骤6:在工具栏单击“ExecuteProgram”(执行程序)按钮,执行2-1.exe程序。运行结果(分行书写。如果运行不成功,则可能的原因是什么?):1)eventcreated2)childcreated3)Pachirledntprwoacietsingonchild4)sbeginning……5)eventsignaled6)parentreceivedtheeventsignalingfromchild7)Parentreleased8)Pressanykeytocontinue这个结果与你期望的一致吗?(从进程并发的角度对结果进行分析)一致阅读和分析程序2-1,请回答:1)程序中,创建一个事件使用了哪一个系统函数?创建时设置的初始信号状态是什么?a.createevent()函数b.FALSE代表无信号状态2)创建一个进程(子进程)使用了哪一个系统函数?CreateChild()函数中的CreateProcess()API函数3)从步骤6的输出结果,对照分析2-1程序,可以看出程序运行的流程吗?请简单描述:Main()检查父进程或子进程是否启动,选择相应入口。然后进入waitforchild(),创建事件,再创建子进程CreateChild()并一直等待直到子进程发出信号。与此同时,子进程完成创建进入SignalParent()函数,打开句柄向父进程传出信号,最后清楚句柄。2.互斥体对象清单2-2的程序中显示的类CCountUpDown使用了一个互斥体来保证对两个线程间单一数值的访问。每个线程都企图获得控制权来改变该数值,然后将该数值写入输出流中。创建者实际上创建的是互斥体对象,计数方法执行等待并释放,为的是共同使用互斥体所需的资源(因而也就是共享资源)。步骤7:在VisualC++窗口的工具栏中单击“打开”按钮,在“打开”对话框中找到并打开实验源程序2-2.cpp。清单2-2利用互斥体保护共享资源//mutex项目#includewindows.h#includeiostream//利用互斥体来保护同时访问的共享资源classCCountUpDown{public://创建者创建两个线程来访问共享值CCountUpDown(intnAccesses):m_hThreadlnc(INVALID_HANDLE_VALUE),m_hThreadDec(INVALID_HANDLE_VALUE),m_hMutexValue(INVALID_HANDLE_VALUE),m_nValue(0),m_nAccess(nAccesses){//创建互斥体用于访问数值m_hMutex