您好,欢迎访问三七文档
实验六、进程控制一、实验目的1.实践并发进程的创建和控制方法。观察和体验进程的动态特性。2.进一步理解进程生命期期间创建、变换、撤销状态变换的过程。3.练习Linux系统中进程创建与控制有关的系统调用的编程和调试技术。二、实验说明1)与进程创建、执行有关的系统调用说明进程可以通过系统调用fork()创建子进程并和其子进程并发执行.子进程初始的执行映像是父进程的一个副本.子进程可以通过exec()系统调用族装入一个新的执行程序。父进程可以使用wait()或waitpid()系统调用等待子进程的结束并负责收集和清理子进程的退出状态。fork()系统调用语法:#includeunistd.hpid_tfork(void);fork成功创建子进程后将返回子进程的进程号,不成功会返回-1.exec系统调用有一组6个函数,其中示例实验中引用了execve系统调用语法:#includeunistd.hintexecve(constchar*path,constchar*argv[],constchar*envp[]);path要装入的新的执行文件的绝对路径名字符串.argv[]要传递给新执行程序的完整的命令参数列表(可以为空).envp[]要传递给新执行程序的完整的环境变量参数列表(可以为空).exec执行成功后将用一个新的程序代替原进程,但进程号不变,它绝不会再返回到调用进程了。如果exec调用失败,它会返回-1。wait()系统调用语法:#includesys/types.h#includesys/wait.hpid_twait(int*status);pid_twaitpid(pid_tpid,int*status,intoption);status用于保留子进程的退出状态pid可以为以下可能值:-1等待所有PGID等于PID的绝对值的子进程1等待所有子进程0等待所有PGID等于调用进程的子进程0等待PID等于pid的子进程option规定了调用waitpid进程的行为:WNOHANG没有子进程时立即返回WUNTRACED没有报告状态的进程时返回wait和waitpid执行成功将返回终止的子进程的进程号,不成功返回-1。getpid()系统调用语法:#includesys/types.h#includeunistd.hpid_tgetpid(void);pid_tgetppid(void);getpid返回当前进程的进程号,getppid返回当前进程父进程的进程号2)与进程控制有关的系统调用说明可以通过信号向一个进程发送消息以控制进程的行为。信号是由中断或异常事件引发的,如:键盘中断、定时器中断、非法内存引用等。信号的名字都以SIG开头,例如SIGTERM、SIGHUP。可以使用kill-l命令查看系统当前的信号集合。信号可在任何时间发生,接收信号的进程可以对接收到的信号采取3种处理措施之一:·忽略这个信号·执行系统默认的处理·捕捉这个信号做自定义的处理信号从产生到被处理所经过的过程:产生(generate)-挂起(pending)-派送(deliver)-部署(disposition)或忽略(igore)一个信号集合是一个C语言的sigset_t数据类型的对象,sigset_t数据类型定义在signal.h中。被一个进程忽略的所有信号的集合称为一个信号掩码(mask)。从程序中向一个进程发送信号有两种方法:调用shell的kill命令,调用kill系统调用函数。kill能够发送除杀死一个进程(SIGKILL、SIGTERM、SIGQUIT)之外的其他信号,例如键盘中断(Ctrl+C)信号SIGINT,进程暂停(Ctrl+Z)信号SIGTSTP等等。调用pause函数会令调用进程的执行挂起直到一个任意信号到来后再继续运行。调用sleep函数会令调用进程的执行挂起睡眠指定的秒数或一个它可以响应的信号到来后继续执行。每个进程都能使用signal函数定义自己的信号处理函数,捕捉并自行处理接收的除SIGSTOP和SIGKILL之外的信号。以下是有关的系统调用的语法说明。kill系统调用语法:#includesys/types.h#includesignal.hintkill(pid_tpid,intsig);pid接收信号的进程号signal要发送的信号kill发送成功返回接收者的进程号,失败返回-1。pause系统调用语法:#includeunistd.hintpause(void);pause挂起调用它的进程直到有任何信号到达。调用进程不自定义处理方法,则进行信号的默认处理。只有进程自定义了信号处理方法捕获并处理了一个信号后,pause才会返回调进程。pause总是返回-1,并设置系统变量errno为EINTR。sleep系统调用语法:#includeunistd.hunsignedintsleep(unsignedintseconds);seconds指定进程睡眠的秒数如果指定的秒数到,sleep返回0。signal系统调用语法为:#includesignal.htypedefvoid(*sighandler_t)(int);sighandler_tsignal(intsignum,sighandler_thandler);signum要捕捉的信号handler进程中自定义的信号处理函数名signal调用成功会返回信号处理函数的返回值,不成功返回-1,并设置系统变量errno为SIG_ERR。三、实验内容与步骤以下实验示例程序应实现一个类似子shell子命令的功能,它可以从执行程序中启动另一个新的子进程并执行一个新的命令和其并发执行.1)打开一终端命令行窗体,新建一个文件夹,在该文件夹中建立以下名为pctl.c的C语言程序:/**Filename:pctl.c*copyright:(C)2006byzhanghonglie*Function:父子进程的并发执行*/#includepctl.hintmain(intargc,char*argv[]){inti;intpid;//存放子进程号intstatus;//存放子进程返回状态char*args[]={/bin/ls,-a,NULL};//子进程要缺省执行的命令signal(SIGINT,(sighandler_t)sigcat);//注册一个本进程处理键盘中断的函数pid=fork();//建立子进程if(pid0)//建立子进程失败?{printf(CreateProcessfail!\n);exit(EXIT_FAILURE);}if(pid==0)//子进程执行代码段{//报告父子进程进程号printf(IamChildprocess%d\nMyfatheris%d\n,getpid(),getppid());pause();//暂停,等待键盘中断信号唤醒//子进程被键盘中断信号唤醒继续执行printf(%dchildwillRunning:\n,getpid());//if(argv[1]!=NULL){//如果在命令行上输入了子进程要执行的命令//则执行输入的命令for(i=1;argv[i]!=NULL;i++)printf(%s,argv[i]);printf(\n);//装入并执行新的程序status=execve(argv[1],&argv[1],NULL);}else{//如果在命令行上没输入子进程要执行的命令//则执行缺省的命令for(i=0;args[i]!=NULL;i++)printf(%s,args[i]);printf(\n);//装入并执行新的程序status=execve(args[0],args,NULL);}}else//父进程执行代码段{printf(\nIamParentprocess%d\n,getpid());//报告父进程进程号if(argv[1]!=NULL){//如果在命令行上输入了子进程要执行的命令//则父进程等待子进程执行结束printf(%dWaitingforchilddone.\n\n);waitpid(pid,&status,0);//等待子进程结束printf(\nMychildexit!status=%d\n\n,status);}else{//如果在命令行上没输入子进程要执行的命令//唤醒子进程,与子进程并发执行不等待子进程执行结束,if(kill(pid,SIGINT)=0)printf(%dWakeup%dchild.\n,getpid(),pid);printf(%ddon'tWaitforchilddone.\n\n,getpid());}}returnEXIT_SUCCESS;}2)再建立以下名为pctl.h的C语言头文件:#includesys/types.h#includewait.h#includeunistd.h#includesignal.h#includestdio.h#includestdlib.h//进程自定义的键盘中断信号处理函数typedefvoid(*sighandler_t)(int);voidsigcat(){printf(%dProcesscontinue\n,getpid());}3)建立以下项目管理文件Makefilehead=pctl.hsrcs=pctl.cobjs=pctl.oopts=-g-call:pctlpctl:$(objs)gcc$(objs)-opctlpctl.o:$(srcs)$(head)gcc$(opts)$(srcs)clean:rmpctl*.o4)输入make命令编译连接生成可执行的pctl程序$gmakegcc-g-cpctl.cgccpctl.o-opctl5)执行pctl程序(注意进程号是动态产生的,每次执行都不相同)$./pctlIamChildprocess4113Myfatheris4112IamParentprocess4112Wakeup4113child.4112don'tWaitforchilddone.4113Processcontinue4113childwillRunning:/bin/ls-a...Makefilepctlpctl.cpctl.hpctl.o$以上程序的输出说明父进程4112创建了一个子进程4113,子进程执行被暂停。父进程向子进程发出键盘中断信号唤醒子进程并与子进程并发执行。父进程并没有等待子进程的结束继续执行先行结束了(此时的子进程成为了孤儿进程,不会有父进程为它清理退出状态了)。而子进程继续执行,它变成了列出当前目录所有文件名的命令ls-a。在完成了列出文件名命令之后,子进程的执行也结束了。此时子进程的退出状态将有初始化进程为它清理。6)再次执行带有子进程指定执行命令的pctl程序:$./pctl/bin/ls-lIamChildprocess4223Myfatheris4222IamParentprocess42224222Waitingforchilddone.可以看到这一次子进程仍然被挂起,而父进程则在等待子进程的完成。为了检测父子进程是否都在并发执行,请输入ctrl+z将当前进程放入后台并输入ps命令查看当前系统进程信息,显示如下:[1]+Stopped./pctl/bin/ls-l$ps-lFSUIDPIDPPIDCPRINIADDRSZWCHANTTYTIMECMD0S0408540830760-1413waitpts/100:00:00bash0T0422240850760-360finishpts/100:00:00pctl1T0422342220760-360finishpts/100:00:00pctl0R0423140850780-1302-pts/100:00:00ps可以看到当前系统中同时有两个叫pctl的进程,它们的进程号分别是4222和4223。它们的状态都为¡
本文标题:实验六进程控制
链接地址:https://www.777doc.com/doc-2459382 .html