步骤3:在“命令提示符”窗口加入参数重新运行生成的可执行文件。运行结果: 范例:E:\\课程\\os课\\os实验\\程序\\os11\\debug>os21 3 (假设编译生成的可执行文件是os21.exe)
按下ctrl+alt+del,调用windows的任务管理器,记录进程相关的行为属性:
11
步骤4:修改清单2-1中的程序,将nClone的定义和初始化方法按程序注释中的修改方法进行修改,编译成可执行文件(执行前请先保存已经完成的工作)。再按步骤2中的方式运行,看看结果会有什么不一样。运行结果:
______Clone_ID:0_______________________________________________________________ 从中你可以得出什么结论:
nClone的作用:__控制程序的执行,当nClone大于等于5时可跳出程序。_______________________________________________________________
_____________________________________________________________________ 变量的定义和初始化方法(位置)对程序的执行结果有影响吗?为什么?________________ ____有,nClone
被修改后会对程序的结束控制产生影
响 _________________________________________________________________________________________________________________________________________________________________________________________________________________________________________
(2). 父子进程的简单通信及终止进程
步骤1:创建一个“Win32 Consol Application”工程,然后拷贝清单2-2中的程序,编译成可执行文件。
步骤2:在VC的工具栏单击“Execute Program”(执行程序) 按钮,或者按Ctrl + F5键,或者在“命令提示符”窗口运行步骤1中生成的可执行文件。运行结果: 范例:E:\\课程\\os课\\os实验\\程序\\os11\\debug>os22 (假设编译生成的可执行文件是os22.exe)
12
步骤3:按源程序中注释中的提示,修改源程序2-2,编译执行(执行前请先保存已经完成的工作)。运行结果:
程序一直执行”Child waiting for suicide instructions.”_
在程序中加入跟踪语句,或调试运行程序,同时参考MSDN中的帮助文件CreateProcess()的使用方法,理解父子进程如何传递参数。给出程序执行过程的大概描述:
_通过main(int argc, char* argv[])传递参数,每次运行时先检测argc的值,若小于1,程序运行结束,否则继续往下执行
步骤4:填空
CreateProcess() 函数有__5______个核心参数?本实验程序中设置的各个参数的值是: a._szFilename___________________________________________; b. ______szCmdLine___________________________________________; c. ______NULL___________________________________________; d. ______NULL___________________________________________; e. ______FALSE___________________________________________;
f. _______CREATE_NEW_CONSOLE__________________________________________; g. _______NULL__________________________________________; h. _______NULL__________________________________________; i._______si__________________________________________; j. _______pi__________________________________________。
步骤5:按源程序中注释中的提示,修改源程序2-2,编译执行。运行结果
__程序一直执行”Creating the child proces”
步骤6:参考MSDN中的帮助文件CreateMutex()、OpenMutex()、ReleaseMutex()和WaitForSingleObject()的使用方法,理解父子进程如何利用互斥体进行同步的。给出父子进程同步过程的一个大概描述:
___ CreateMutex()创建互斥体,OpenMutex()打开互斥体,ReleaseMutex()释放互斥体,WaitForSingleObject()检测hHandle事件的信号状态,通过这些方法可实现当前只有一个
13
进程被创建或使用,实现进程的同步。
____________________________________________________________________________
3、 实验结论
通过对进程的操作,如创建进程,实现对进程的简单控制。创建互斥体,解决了进程的同步问题,两者相互使用,使进程的运行情况得到了很好的管理。
4、 程序清单
清单2-1 创建子进程 // proccreate项目
#include
// 创建传递过来的进程的克隆过程并赋于其ID值 void StartClone(int nCloneID) {
// 提取用于当前可执行文件的文件名 TCHAR szFilename[MAX_PATH] ;
GetModuleFileName(NULL, szFilename, MAX_PATH) ;
// 格式化用于子进程的命令行并通知其EXE文件名和克隆ID TCHAR szCmdLine[MAX_PATH]; sprintf(szCmdLine,\ // 用于子进程的STARTUPINFO结构 STARTUPINFO si;
ZeroMemory(&si , sizeof(si) ) ; si.cb = sizeof(si) ; // 必须是本结构的大小
// 返回的用于子进程的进程信息 PROCESS_INFORMATION pi;
// 利用同样的可执行文件和命令行创建进程,并赋于其子进程的性质 BOOL bCreateOK=::CreateProcess( szFilename, // 产生这个EXE的应用程序的名称 szCmdLine, // 告诉其行为像一个子进程的标志 NULL, // 缺省的进程安全性 NULL, // 缺省的线程安全性 FALSE, // 不继承句柄 CREATE_NEW_CONSOLE, // 使用新的控制台 NULL, // 新的环境 NULL, // 当前目录 &si, // 启动信息 &pi) ; // 返回的进程信息
// 对子进程释放引用
14
if (bCreateOK) {
CloseHandle(pi.hProcess) ; CloseHandle(pi.hThread) ; } }
int main(int argc, char* argv[] ) {
// 确定派生出几个进程,及派生进程在进程列表中的位置 int nClone=0;
//修改语句:int nClone;
//第一次修改:nClone=0; if (argc > 1) {
// 从第二个参数中提取克隆ID :: sscanf(argv[1] , \ }
//第二次修改:nClone=0; // 显示进程位置
std :: cout << \ << \ << std :: endl; // 检查是否有创建子进程的需要 const int c_nCloneMax=5; if (nClone < c_nCloneMax) { // 发送新进程的命令行和克隆号 StartClone(++nClone) ; }
// 等待响应键盘输入结束进程 getchar();
return 0; }
清单2-2 父子进程的简单通信及终止进程的示例程序 // procterm项目
# include
static LPCTSTR g_szMutexName = \
// 创建当前进程的克隆进程的简单方法 void StartClone() {
// 提取当前可执行文件的文件名 TCHAR szFilename[MAX_PATH] ;
15