孤儿进程
一个父进程退出,而它的一个或多个子进程还在运行,那么那些子进程将成为孤儿进程。孤儿进程将被init进程(进程号为1)所收养,并由init进程对它们完成状态收集工作。
当一个孤儿进程被一号进程收养的时候,一号进程会代替它的父进程管理它,在它完成自己的声明周期后,一号进程会结束掉孤儿进程,所以没有危害。
孤儿进程实现代码:
#include<stdio.h>
#include<stdlib.h>
#include<unistd.h>
#include<error.h>
int main(void)
{
pid_t pid;
pid = fork();
if(pid < 0){
perror("fork error");
return -1;
}else if(pid == 0)//子进程
{
printf("i am the child process\n");
printf("pid = %d, ppid = %d\n", getpid(), getppid());
while(1){
sleep(1);
}
}else//父进程
{
printf("i am the parent process\n");
printf("pid = %d, ppid = %d\n", getpid(), getppid());
sleep(3);
}
}
这里可以看到子进程3022的父进程本来是3021,但是由于父进程退出,子进程没有退出,子进程变成了孤儿进程继续执行,并且由1号进程领养,成为了1号进程的子进程。
僵尸进程
一个进程使用fork创建子进程,如果子进程退出,而父进程并没有调用wait或waitpid获取子进程的状态信息,那么子进程的进程描述符仍然保存在系统中。这种进程称之为僵死进程。
如果进程不调用wait / waitpid的话, 那么保留的那段信息就不会释放,其进程号就会一直被占用,但是系统所能使用的进程号是有限的,如果大量的产生僵死进程,将因为没有可用的进程号而导致系统不能产生新的进程. 此即为僵尸进程的危害,应当避免。
使用ps -A -o stat,ppid,pid,cmd |grep a.out命令可以查看a.out生成的僵尸进程,stat状态为z或Z的就是僵尸进程,这时候需要kill pid杀死僵尸进程。
实现代码如下:
#include<stdio.h>
#include<stdlib.h>
#include<unistd.h>
#include<error.h>
#include<sys/wait.h>
int main(void)
{
pid_t pid;
pid = fork();
if(pid < 0){
perror("fork error");
return -1;
}else if(pid == 0)//子进程
{
printf("i am the child process\n");
printf("pid = %d, ppid = %d\n", getpid(), getppid());
}else//父进程
{
//wait(NULL);
printf("i am the parent process\n");
printf("pid = %d, ppid = %d\n", getpid(), getppid());
while(1){
;
}
}
}
此时进程3337就成为了僵尸进程,需要用kill 3337杀死僵尸进程。