wait()函数
当fork启动一个子进程之后,有时,我们想知道子进程何时结束,因此,我们可以通过调用wait()函数让父进程等待子进程的结束。
函数原型为
#include<sys/types.h>
#include<sys/weait.h>
pid_t wait(int* stat_loc);
wait系统调用将暂停父进程直到它的子进程结束为止。此调用返回的时子进程的PID,通常是已经结束的子进程的PID。状态信息允许父进程了解子进程的退出状态,即子进程main函数返回的值或exit函数的退出码。如果stat_loc不是空指针的话,状态信息将被写入它所指向的位置,我们可以用sys/wait.h文件中定义的宏来解释状态信息,如下图示:
#include<stdio.h>
#include<assert.h>
#include<stdlib.h>
#include<sys/types.h>
#include<unistd.h>
#include<sys/wait.h>
int main()
{
pid_t pid;
char* message;
int n;
printf("fork program starting!\n");
pid = fork();
assert(pid != -1);
if (pid == 0)
{
message = "child";
n = 3;
}
else
{
message = "father";
n = 7;
int val;
pid_t _pid = wait(&val);
if (WIFEXITED(val)) //如果子进程正常结束,它就取一个非零值
{
printf("val = %d\n",WEXITSTATUS(val));//如果WIFEXITED非零,他返回子进程的退出码
}
}
for (int i = 0; i<n; i++)
{
printf("%s\n",message);
//printf("cur_pid=%d,cur_ppid=%d,message=%s,n=%d\n",getpid(),getppid(),message,n);
sleep(1);
}
exit(3);//退出码设置为3
}
父进程用wait系统调用将自己的执行挂起,直到子进程的状态信息出现为止,这将发生在子进程调用exit的时候。我们将退出码设置为3。父进程然后继续执行,通过测试wait调用的返回值来判断子进程是否终止。如果是,就从状态信息中提取出子进程的退出码。
僵尸进程
用fork来创建进程确实很实用,但你必须清楚子进程的运行状况,子进程中终止时,与父进程的关联还保持,直到父进程正常终止或父进程调用wait才告结束。因此,进程表中代表子进程的表项不会立即释放,虽然子进程已经结束,但他仍然存在于系统中,因为他的退出码还需要保存起来,以备父进程调用wait使用。而这时他将会成为一个死进程/僵尸进程。
简单地说就是:子进程先于父进程结束,父进程没有调用wait获取子进程的退出码,子进程变成僵死进程
解决方案:
- 父进程调用wait();
- 如果父进程结束,那么init会成为子进程的父进程,然后init调用wait获取退出码,子进程便不会变成僵尸进程。
代码实现为上面的代码。
还有一个系统调用是用来等待子进程的结束,是waitpid()函数。
函数原型为
#include<sys/types.h>
#include<sys/weait.h>
pid_t waitpid(pid_t pid, int* stat_loc, int option);
pid指定的是需要等待的子进程的pid,如果stat_loc不是空指针的话,状态信息将被写入它所指向的位置,option参数用来改变waitpid的行为,其中最常用的是WNOHANG,它的作用是防止waitpid调用将调用者的执行挂起。
因此,如果想让父进程周期性的检查某个特定的子进程是否已终止,可以使用如下调用方式;
waitpid(child_pid, (int)* 0, WNOHANG);
如果子进程没有结束或意外终止,它就返回0,否则返回child_pid。如果waitpid失败,将返回-1并设置error。
替换进程映像exec系列函数
exec函数可以把当前进程替换为一个新进程,新进程由path或file参数指定。可以使用exec函数将程序的执行从一个程序切换到另一个程序
详细介绍请看这个博客替换进程映像exec系列函数,写的挺不错的。
信号
此处内容日后有机会再补充。。。