操作系统实验进程控制Word格式文档下载.docx
- 文档编号:19072952
- 上传时间:2023-01-03
- 格式:DOCX
- 页数:8
- 大小:23.42KB
操作系统实验进程控制Word格式文档下载.docx
《操作系统实验进程控制Word格式文档下载.docx》由会员分享,可在线阅读,更多相关《操作系统实验进程控制Word格式文档下载.docx(8页珍藏版)》请在冰豆网上搜索。
wait()系统调用语法:
sys/types.h>
sys/wait.h>
pid_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()系统调用语法:
pid_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系统调用语法:
intkill(pid_tpid,intsig);
pid接收信号的进程号
signal要发送的信号
kill发送成功返回接收者的进程号,失败返回-1。
pause系统调用语法:
#include<
intpause(void);
pause挂起调用它的进程直到有任何信号到达。
调用进程不自定义处理方法,则进行信号的默认处理。
只有进程自定义了信号处理方法捕获并处理了一个信号后,pause才会返回调进程。
pause总是返回-1,并设置系统变量errno为EINTR。
sleep系统调用语法:
unsignedintsleep(unsignedintseconds);
seconds指定进程睡眠的秒数如果指定的秒数到,sleep返回0。
signal系统调用语法为:
typedefvoid(*sighandler_t)(int);
sighandler_tsignal(intsignum,sighandler_thandler);
signum要捕捉的信号handler进程中自定义的信号处理函数名signal调用成功会返回信号处理函数的返回值,不成功返回-1,并设置系统变量errno为SIG_ERR。
1.3示例实验
以下实验示例程序应实现一个类似子shell子命令的功能,它可以从执行程序中启动另一个新的子进程并执行一个新的命令和其并发执行.
1)打开一终端命令行窗体,新建一个文件夹,在该文件夹中建立以下名为pctl.c的C语言程序:
/*
*:
pctl.c
*copyright:
(C)2006byzhanghonglie
*Function:
父子进程的并发执行
*/
#include"
pctl.h"
intmain(intargc,char*argv[])
{inti;
intpid;
//存放子进程号intstatus;
//存放子进程返回状态char*args[]={"
/bin/ls"
"
-a"
NULL};
//子进程要缺省执行的命令signal(SIGINT,(sighandler_t)sigcat);
//注册一个本进程处理键盘中断的函数pid=fork();
//建立子进程if(pid<
0)//建立子进程失败?
{
printf("
CreateProcessfail!
\n"
);
exit(EXIT_FAILURE);
}
if(pid==0)//子进程执行代码段
//报告父子进程进程号
IamChildprocess%d\nMyfatheris%d\n"
getpid(),getppid());
pause();
//暂停,等待键盘中断信号唤醒
//子进程被键盘中断信号唤醒继续执行
%dchildwillRunning:
\n"
getpid());
//if(argv[1]!
=NULL){
//如果在命令行上输入了子进程要执行的命令
//则执行输入的命令
for(i=1;
argv[i]!
=NULL;
i++)printf("
%s"
argv[i]);
printf("
//装入并执行新的程序
status=execve(argv[1],&
argv[1],NULL);
}else{
//如果在命令行上没输入子进程要执行的命令
//则执行缺省的命令
for(i=0;
args[i]!
args[i]);
status=execve(args[0],args,NULL);
else//父进程执行代码段
{
\nIamParentprocess%d\n"
//报告父进程进程号if(argv[1]!
//如果在命令行上输入了子进程要执行的命令
//则父进程等待子进程执行结束
%dWaitingforchilddone.\n\n"
pid);
waitpid(pid,&
status,0);
//等待子进程结束
\nMychildexit!
status=%d\n\n"
status);
//唤醒子进程,与子进程并发执行不等待子进程执行结束,
if(kill(pid,SIGINT)>
=0)
%dWakeup%dchild.\n"
getpid(),pid);
%ddon'
tWaitforchilddone.\n\n"
}}
returnEXIT_SUCCESS;
}
2)再建立以下名为pctl.h的C语言头文件:
wait.h>
stdio.h>
stdlib.h>
//进程自定义的键盘中断信号处理函数
typedefvoid(*sighandler_t)(int);
voidsigcat(){
%dProcesscontinue\n"
3)建立以下项目管理文件Makefile
head=pctl.hsrcs=pctl.cobjs=pctl.oopts=-g-c
all:
pctl
pctl:
$(objs)gcc$(objs)-opctlpctl.o:
$(srcs)$(head)gcc$(opts)$(srcs)clean:
rmpctl*.o
4)输入make命令编译连接生成可执行的pctl程序
$gmake
gcc-g-cpctl.c
gccpctl.o-opctl
5)执行pctl程序(注意进程号是动态产生的,每次执行都不相同)
$./pctl
IamChildprocess4113
Myfatheris4112
IamParentprocess4112Wakeup4113child.
4112don'
tWaitforchilddone.
4113Processcontinue
4113childwillRunning:
/bin/ls-a
...Makefilepctlpctl.cpctl.hpctl.o
$
以上程序的输出说明父进程4112创建了一个子进程4113,子进程执行被暂停。
父进程向子进程发出键盘中断信号唤醒子进程并与子进程并发执行。
父进程并没有等待子进程的结束继续执行先行结束了(此时的子进程成为了孤儿进程,不会有父进程为它清理退出状态了)。
而子进程继续执行,它变成了列出当前目录所有文件名的命令ls-a。
在完成了列出文件名命令之后,子进程的执行也结束了。
此时子进程的退出状态将有初始化进程为它清理。
6)再次执行带有子进程指定执行命令的pctl程序:
$./pctl/bin/ls-l
IamChildprocess4223
Myfatheris4222
IamParentprocess4222
4222Waitingforchilddone.
可以看到这一次子进程仍然被挂起,而父进程则在等待子进程的完成。
为了检测父子进程是否都在并发执行,请输入ctrl+z将当前进程放入后台并输入ps命令查看当前系统进程信息,显示如下:
[1]+Stopped./pctl/bin/ls-l$ps-l
FSUIDPIDPPIDCPRINIADDRSZWCHANTTYTIMECMD
0S0408540830760-1413waitpts/100:
00:
00bash
1T0422240850760-360finishpts/100:
00pctl
2T0422342220760-360finishpts/100:
0R0423140850780-1302-pts/100:
00ps
可以看到当前系统中同时有两个叫pctl的进程,它们的进程号分别是4222和4223。
它们的状态都为―T‖,说明当前都被挂起。
4223的父进程是4222,而4222的父进程是4085,也就是bash-shell。
为了让pctl父子进程继续执行,请输入fg命令让pctl再次返回前台,显示如下:
$fg
./pctl/bin/ls-l
现在pctl父子进程从新返回前台。
我们可以通过键盘发键盘中断信号来唤醒
pctl父子进程继续执行,输入ctrl+c,将会显示:
4222Processcontinue
4223Processcontinue
4223childwillRunning:
/bin/ls-ltotal1708
-rw-r--r--1rootroot176May811:
11Makefile
-rwxr-xr-x1rootroot8095May814:
08pctl
-rw-r--r--1rootroot2171May814:
08pctl.c-rw-r--r--1rootroot269May811:
10pctl.h
-rw-r--r--1rootroot4156May814:
08pctl.o
Mychildexit!
status=0
以上输出说明了子进程在捕捉到键盘中断信号后继续执行了指定的命令,按我们要求的长格式列出了当前目录中的文件名,父进程在接收到子进程执行结束的信号后将清理子进程的退出状态并继续执行,它报告了子进程
的退出编码(0表示子进程正常结束)最后父进程也结束执行。
1.4实验结果
参考以上示例程序中建立并发进程的方法,编写一个多进程并发执行程序。
父进程首先创建一个执行ls命令的子进程然后再创建一个执行ps命令的子进程,并控制
ps命令总在ls命令之前执行。
1.5.实验小结
第一次实验的操作是比较简单的,通过老师的讲解,很快就能掌握要点,但是在执行过程中,很容易打错字符或漏掉标点符号,导致实验结果不正确,通过不断查找与更改,最后得出正确结论。
此次实验很需要耐心和毅力,能不断加强我们的动手能力后调试能力。
- 配套讲稿:
如PPT文件的首页显示word图标,表示该PPT已包含配套word讲稿。双击word图标可打开word文档。
- 特殊限制:
部分文档作品中含有的国旗、国徽等图片,仅作为作品整体效果示例展示,禁止商用。设计者仅对作品中独创性部分享有著作权。
- 关 键 词:
- 操作系统 实验 进程 控制