C语言中的wait系统调用详细指南

先决条件: 货叉系统调用
对wait()的调用将阻止调用进程, 直到其子进程之一退出或接收到信号为止。子进程终止后, 父进程
继续
等待系统调用指令后执行。
子进程可能由于以下原因而终止:

  • 它调用exit();
  • 它从main返回(int)
  • 它从操作系统或其他进程接收信号, 其默认操作是终止。
C语言中的wait系统调用详细指南

文章图片
C语言语法:
#include#include// take one argument status and returns // a process ID of dead children.pid_t wait(int *stat_loc);

如果任何进程具有多个子进程, 则在调用wait()之后, 如果没有子进程终止, 则父进程必须处于等待状态。
如果只有一个子进程终止, 则返回一个wait()返回终止的子进程的进程ID。
如果终止了多个子进程, 则比wait()收获任何子进程
随便的孩子
并返回该子进程的进程ID。
当wait()返回时, 它们还定义
退出状态
(告诉我们, 一个进程为什么终止), 如果没有, 则通过指针
null
.
如果任何进程没有子进程, 则wait()立即返回" -1"。
注意:"由于环境问题, 此代码无法在简单的IDE中运行, 因此请使用终端来运行代码"
例子:
// C program to demonstrate working of wait() #include< stdio.h> #include< stdlib.h> #include< sys/wait.h> #include< unistd.h> int main() { pid_t cpid; if (fork()== 0) exit (0); /* terminate child */ else cpid = wait(NULL); /* reaping parent */ printf ( "Parent pid = %d\n" , getpid()); printf ( "Child pid = %d\n" , cpid); return 0; }

输出如下:
Parent pid = 12345678 Child pid = 89546848

// C program to demonstrate working of wait() #include< stdio.h> #include< sys/wait.h> #include< unistd.h> int main() { if (fork()== 0) printf ( "HC: hello from child\n" ); else { printf ( "HP: hello from parent\n" ); wait(NULL); printf ( "CT: child has terminated\n" ); }printf ( "Bye\n" ); return 0; }

输出:取决于环境
HC: hello from childByeHP: hello from parentCT: child has terminated(or)HP: hello from parentHC: hello from childCT: child has terminated// this sentence does // not print before HC // because of wait.Bye

子状态信息:
通过等待报告的关于孩子的状态信息不仅是孩子的退出状态, 还包括
  • 正常/异常终止
  • 终止原因
  • 退出状态
要查找有关状态的信息, 我们使用
WIF
…。宏
1.
WIFEXITED(状态)
:孩子正常退出
?
【C语言中的wait系统调用详细指南】WEXITSTATUS(状态)
:孩子退出时返回代码
2.
WIFSIGNALED(状态)
:孩子退出, 因为未捕获到信号
?
WTERMSIG(状态)
:给出终止信号的编号
3.
WIFSTOPPED(状态)
:孩子停止了
?
WSTOPSIG(状态)
:给出停止信号的编号
/*if we want to prints information about a signal */void psignal(unsigned sig, const char *s);

例子:
检查以下程序的输出。
// C program to demonstrate working of status // from wait. #include< stdio.h> #include< stdlib.h> #include< sys/wait.h> #include< unistd.h> void waitexample() { int stat; // This status 1 is reported by WEXITSTATUS if (fork() == 0) exit (1); else wait(& stat); if (WIFEXITED(stat)) printf ( "Exit status: %d\n" , WEXITSTATUS(stat)); else if (WIFSIGNALED(stat)) psignal(WTERMSIG(stat), "Exit signal" ); }// Driver code int main() { waitexample(); return 0; }

输出如下:
Exit status: 1

我们知道如果终止了多个子进程, 那么wait()会获得任意子进程, 但是如果我们想获得任何特定的子进程, 我们将使用waitpid()功能。
C语言语法:pid_t waitpid(child_pid, &status, 选项);
选项参数
  • 如果为0表示没有选项, 则父级不必等待终止子级。
  • If万航表示如果子项未终止则父项不等待, 只需检查并返回waitpid()。(不阻止父级进程)
  • 如果child_pid是-1那么意味着随便的孩子, 这里的waitpid()工作与wait()工作相同。
waitpid()的返回值
  • 孩子的pid(如果孩子退出了)
  • 0, 如果使用WNOHANG并且未退出孩子
// C program to demonstrate waitpid() #include< stdio.h> #include< stdlib.h> #include< sys/wait.h> #include< unistd.h> void waitexample() { int i, stat; pid_t pid[5]; for (i=0; i< 5; i++) { if ((pid[i] = fork()) == 0) { sleep(1); exit (100 + i); } }// Using waitpid() and printing exit status // of children. for (i=0; i< 5; i++) { pid_t cpid = waitpid(pid[i], & stat, 0); if (WIFEXITED(stat)) printf ( "Child %d terminated with status: %d\n" , cpid, WEXITSTATUS(stat)); } }// Driver code int main() { waitexample(); return 0; }

输出如下:
Child 50 terminated with status: 100Child 51 terminated with status: 101Child 52 terminated with status: 102Child 53 terminated with status: 103Child 54 terminated with status: 104

这里, Children pids取决于系统, 但是为了打印所有的孩子信息。
如果发现任何不正确的地方, 或者想分享有关上述主题的更多信息, 请写评论。

    推荐阅读