所谓孤儿进程就是一个父进程退出,而它的一个或多个子进程还在运行,那么那些子进程将成为孤儿进程。孤儿进程将被init进程(进程号为1)所收养,并由init进程对它们完成状态收集工作。
由于孤儿进程会被init进程给收养,所以孤儿进程不会对系统造成危害。以下是孤儿进程的代码
#include<stdio.h>
#include<unistd.h>
#include<sys/types.h>
#include<string.h>
#include<fcntl.h>
#include<sys/stat.h>
#include<sys/wait.h>
int main(int argc, const char *argv[])
{
pid_t pid = fork(); //创建子进程
if(pid > 0)
{
printf("这是父进程\n\n");
while(1)
{
printf("父进程 %d,子%d 执行并结束退出\n",getpid(),pid);
break;
}
}
else if(0 == pid)
{
while(1)
{
sleep(1);
printf("子进程%d %d变成孤儿进程\n",getppid(),pid); //等子1进程执行是父进程已经结束
}
}
return 0;
}
所谓僵尸进程就是在fork()/execve()过程中,假设子进程结束时父进程仍存在,而父进程fork()之前既没安装SIGCHLD信号处理函数调用waitpid()等待子进程结束,又没有显式忽略该信号,则子进程成为僵死进程,无法正常结束,此时即使是root身份kill -9也不能杀死僵死进程。补救办法是杀死僵尸进程的父进程(僵死进程的父进程必然存在),僵死进程成为”孤儿进程”,过继给1号进程init,init始终会负责清理僵死进程。我们知道在unix/linux中,正常情况下,子进程是通过父进程创建的,子进程在创建新的进程。子进程的结束和父进程的运行是一个异步过程,即父进程永远无法预测子进程 到底什么时候结束。 当一个 进程完成它的工作终止之后,它的父进程需要调用wait()或者waitpid()系统调用取得子进程的终止状态。僵尸进程是有危害的。以下为代码段
#include<stdio.h>
#include<unistd.h>
#include<sys/types.h>
#include<string.h>
#include<fcntl.h>
#include<sys/stat.h>
#include<sys/wait.h>
int main(int argc, const char *argv[])
{
pid_t pid = fork();
if(pid>0)
{
printf("这是父进程\n\n");
//阻塞等待子进程退出,并回收子进程资源
//pid_t cpid = wait(NULL);
/* int wstatus = 0;
pid_t cpid = wait(&wstatus);
printf("cpid = %d wstatus = %d\n",cpid ,wstatus);
int res=WEXITSTATUS(wstatus); //通过宏函数在wstatus中获取exit中的值
printf("res=%d\n",res);
if(WIFEXITED(wstatus))
printf("cpid = %d 正常退出\n",cpid);
else
printf("cpid = %d 异常退出\n",cpid);*/
while(1)
{
printf("这是父进程%d ,子:%d\n",getpid(),pid);
sleep(1);
}
}
else if(0==pid)
{
int i=0;
while(1)
{
printf("这是子进程%d,子%d\n",getppid(),getpid());
if(i==3)
{
printf("子进程退出\n");
_exit(0);
}
i++;
sleep(1);
}
}
else
{
perror("fork");
return -1;
}
return 0;
}