Linux 系统进程控制编程 ----wait 函数使用
这篇文章主要介绍 wait 和类似的 waitpid 调用 。
当进程终止时 , 会向其父进程发送 SIGCHLD 信号 , 这个异步事件可以在父进程运行的任何时候发生 , 包括正常和异常终止两种 。 调用 wait 和 waitpid 的进程可能会有以下三种情况 :
1 )阻塞: 如果其所有的子进程都还在运行
2) 带子进程的终止状态正常返回 : 其中一个子进程终止
3 )出错返回 : 没有子进程
wait( 等待子进程中断或结束 )
表头文件
#include
#include
定义函数 pid_t wait (int * status);
函数说明
wait() 会暂时停止目前进程的执行,直到有信号来到或子进程结束。如果在调用 wait() 时子进程已经结束,则 wait() 会立即返回子进程结束状态值。子进程的结束状态值会由参数 status 返回,而子进程的进程识别码也会一快返回。如果不在意结束状态值,则参数 status 可以设成 NULL 。子进程的结束状态值请参考 waitpid( ) 。
如果执行成功则返回子进程识别码 (PID) ,如果有错误发生则返回,返回值 -1 。失败原因存于 errno 中。
waitpid( 等待子进程中断或结束 )
表头文件
#include
#include
定义函数 pid_t waitpid( pid_t pid, int * status, int options );
函数说明
waitpid() 会暂时停止目前进程的执行 , 直到有信号来到或子进程结束。如果在调用 wait() 时子进程已经结束 , 则 wait() 会立即返回子进程结束状态值。子进程的结束状态值会由参数 status 返回 , 而子进程的进程识别码也会一并返回。如果不在意结束状态值 , 则参数 status 可以设成 NULL 。参数 pid 为欲等待的子进程识别码 ,
其他数值意义如下 :
pid<-1 等待进程组识别码为 pid 绝对值的任何子进程。
pid=-1 等待任何子进程 , 相当于 wait() 。
pid= 0 等待进程组识别码与目前进程相同的任何子进程。
pid> 0 等待任何子进程识别码为 pid 的子进程。
参数 option 可以为 0 (不使用时) 或下面的 OR ( ” | “ ) 组合 :
WNOHANG : 如果没有任何已经结束的子进程则马上返回 , 不予以等待。
WUNTRACED: 如果子进程进入暂停执行情况则马上返回 , 但结束状态不予以理会。子进程的结束状态返回后存于 status, 底下有几个宏可判别结束情况 :
如果执行成功则返回子进程识别码 (PID) , 如果有错误发生则返回,返回值 -1 。失败原因存于 errno 中。
/**************************************
******waitpid.c - Simple wait usage
**************************************/
#include
#include
#include
#include
#include
int main( void )
{
pid_t childpid;
int status;
childpid = fork();
if ( -1 == childpid )
{
perror( "fork()" );
exit( EXIT_FAILURE );
}
else if ( 0 == childpid )
{ // 子进程中
puts( "In child process" );
sleep( 3 ); // 让子进程睡眠 3 秒,看看父进程的行为
printf("/t child pid = %d /n", getpid());
printf("/t child ppid = %d /n", getppid());
exit(EXIT_SUCCESS);
}
else
{ // 父进程中
waitpid( childpid, &status, 0 );
puts( "in parent" );
printf( "/tparent pid = %d/n", getpid() );
printf( "/tparent ppid = %d/n", getppid() );
printf( "/tchild process exited with status %d /n", status );
}
exit(EXIT_SUCCESS);
}
[root@localhost src]# gcc waitpid.c
[root@localhost src]# ./a.out
In child process
child pid = 4469
child ppid = 4468
in parent
parent pid = 4468
parent ppid = 4379
child process exited with status 0
[root@localhost src]#
如果将上面 “waitpid( childpid, &status, 0 );” 行注释掉,程序执行效果如下:
[root@localhost src]# ./a.out
In child process
in parent
parent pid = 4481
parent ppid = 4379
child process exited with status 1331234400
[root@localhost src]# child pid = 4482
child ppid = 1
子进程还没有退出,父进程已经退出了。
waitpid 和 wait 的区别 :
waitpid 提供了 wait 函数不能实现的 3 个功能 :
1 ) waitpid 等待特定的子进程 , 而 wait 则返回任一终止状态的子进程(第一个结束的进程) ;
2 ) waitpid 提供了一个 wait 的非阻塞版本 ;
3 ) waitpid 支持作业控制 ( 以 WUNTRACED 选项 ).
用于检查 wait 和 waitpid 两个函数返回终止状态的宏 :
这两个函数返回的子进程状态都保存在 statloc 指针中 , 用以下 3 个宏可以检查该状态 :
WIFEXITED(status): 若为正常终止 , 则为真 . 此时可执行
WEXITSTATUS(status): 取子进程传送给 exit 或 _exit 参数的低 8 位
WIFSIGNALED(status): 若为异常终止 , 则为真 。 此时可执行
WTERMSIG(status): 取使子进程终止的信号编号
WIFSTOPPED(status): 若为当前暂停子进程 , 则为真 。 此时可执行
WSTOPSIG(status): 取使子进程暂停的信号编号 .