linux中wait()系统调⽤的例⼦,linux中wait系统调⽤
系统中的僵⼫进程都要由wait系统调⽤来回收,下⾯就通过实战看⼀看wait的具体⽤法:
wait的函数原型是:
#include /* 提供类型pid_t的定义 */
#include
pid_t wait(int *status);
返回值: 如果执⾏成功则返回⼦进程识别码(PID),如果有错误发⽣则返回-1。失败原因存于errno中。
进程⼀旦调⽤了wait,就⽴即阻塞⾃⼰,由wait⾃动分析是否当前进程的某个⼦进程已经 退出,如果让它到了这样⼀个已经变成僵⼫的⼦进程, wait就会收集这个⼦进程的信息,并把它彻底销毁后返回;如果没有到这样⼀个⼦进程,wait就会⼀直阻塞在这⾥,直到有⼀个出现为⽌。
参数status⽤来保存被收集进程退出时的⼀些状态,它是⼀个指向int类型的指针。但如果我们对这个⼦进程是如何死掉的毫不在意,只想把这个僵⼫进程消灭掉,(事实上绝⼤多数情况下,我们都会这样想),我们就可以设定这个参数为NULL,就象下⾯这样:
pid = wait(NULL);
如果成功,wait会返回被收集的⼦进程的进程ID,如果调⽤进程没有⼦进程,调⽤就会失败,此时wait返回-1,同时errno被置为ECHILD。
下⾯就让我们⽤⼀个例⼦来实战应⽤⼀下wait调⽤:
/* wait1.c */
#include
#include
#include
#include
#include
#include
int main()
{
pid_t pc, pr;
pc = fork();
if ( pc < 0 ) /* 如果出错 */
{
printf("create child prcocess error: %s/n", strerror(errno));
exit(1);
}
else if ( pc == 0) /* 如果是⼦进程 */
{
printf("I am child process with pid %d /n", getpid());
sleep(3);/* 睡眠3秒钟 */
exit(0);
}
else /* 如果是⽗进程 */
{
printf("Now in parent process, pid = %d/n", getpid());
printf("I am waiting child process to exit./n");
pr = wait(NULL); /* 在这⾥等待⼦进程结束 */
if ( pr > 0 ) /*⼦进程正常返回*/
printf("I catched a child process with pid of %d/n", pr);
else /*出错*/
printf("error: %s/n./n", strerror(errno));
}
exit(0);
}
编译并运⾏:
$ gcc wait1.c -o wait1
$ ./wait1
I am child process with pid 2351
Now in parent process, pid = 2350
I am waiting child process to exit.
I catched a child process with pid of 2351
可以明显注意到,在第2⾏结果打印出来前有3秒钟的等待时间,这就是我们设定的让⼦进程睡 眠的时间,只有⼦进程从睡眠中苏醒过来,它才能正常退 出,也就才能被⽗进程捕捉到。其实这⾥我们不管设定⼦进程睡眠的时间有多长,⽗进程都会⼀直等待下去,读者如果有兴趣的话,可以试着⾃⼰修改⼀下这个数 值,看看会出现怎样的结果。
如果参数status的值不是NULL,wait就会把⼦进程退出时的状态取出并存⼊其中, 这是⼀个整数值(int),指出了⼦进程是正常退出还是
被⾮正常结束的(⼀个进程也可以被其他进程⽤信号结束,我们将在以后的⽂章中介绍),以及正常结束时的返回值,或被哪⼀个信号结束的等信息。由于这些信息 被存放在⼀个整数的不同⼆进制位中,所以⽤常规的⽅法读取会⾮常⿇烦,⼈们就设计了⼀套专门的宏(macro)来完成这项⼯作,下⾯我们来学习⼀下其中最常⽤的两个:
1,WIFEXITED(status) 这个宏⽤来指出⼦进程是否为正常退出的,如果是,它会返回⼀个⾮零值。
(请注意,虽然名字⼀样,这⾥的参数status并不同于wait唯⼀的参数–指向整数的指针status,⽽是那个指针所指向的整数,切记不要搞混了。)
2, WEXITSTATUS(status) 当WIFEXITED返回⾮零值时,我们可以⽤这个宏来提取⼦进程的返回值,如果⼦进程调⽤exit(5)退
出,WEXITSTATUS(status) 就会返回5;如果⼦进程调⽤exit(7),WEXITSTATUS(status)就会返回7。请注意,如果进程不是正常退出的,也就是说, WIFEXITED返回0,这个值就毫⽆意义。
下⾯通过例⼦来实战⼀下我们刚刚学到的内容:
/* wait2.c */
#include
#include
#include
int main()
{
int status;
pid_t pc, pr;
pc = fork();
if ( pc < 0) /* 如果出错 */
printf("error occured./n");
else if ( pc == 0 ) /* ⼦进程 */
{
printf("This is child process with pid of %d./n", getpid()); exit(3); /* ⼦进程返回3 */
}
else /* ⽗进程 */
{
pr = wait(&status);
if ( WIFEXITED(status) ) /* 如果WIFEXITED返回⾮零值 */ {
printf("The child process %d exit normally./n", pr);
printf("the return code is %d./n", WEXITSTATUS(status)); }
else /* 如果WIFEXITED返回零 */
printf("The child process %d exit abnormally./n", pr);
linux下的sleep函数}
exit(0);
}
编译并运⾏:
$ gcc wait2.c -o wait2
$ ./wait2
This is child process with pid of 1538.
the child process 1538 exit normally.
the return code is 3.
⽗进程准确捕捉到了⼦进程的返回值3,并把它打印了出来。
当然,处理进程退出状态的宏并不⽌这两个,但它们当中的绝⼤部分在平时的编程中很儆玫剑 鸵膊辉谡饫锢朔哑 樯芰耍 ⾏巳さ亩琳呖梢宰约翰卧腖inux man pages去了解它们的⽤法。
版权声明:本站内容均来自互联网,仅供演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系QQ:729038198,我们将在24小时内删除。
发表评论