目录
waitpid等待
进程树
孤儿进程
waitpid等待
Linux中父进程除了可以启动子进程,还要负责回收子进程的状态。如果子进程结束后父进程没有正常回收,那么子进程就会变成一个僵尸进程——即程序执行完成,但是进程没有完全结束,其内核中PCB结构体(下文介绍)没有释放。在上面的例子中,父进程在子进程结束前就结束了,那么其子进程的回收工作就交给了父进程的父进程的父进程
#include <sys/types.h>
#include <sys/wait.h>
/** 等待子进程的终止并获取子进程的退出状态
* 功能简单 没有选择
*/
pid_t wait(int *wstatus);
/**
* 功能灵活 可以设置不同的模式 可以等待特定的子进程
*
* pid: 等待的模式
* (1) 小于-1 例如 -1 * pgid,则等待进程组ID等于pgid的所有进程终止
* (2) 等于-1 会等待任何子进程终止,并返回最先终止的那个子进程的进程ID -> 儿孙都算
* (3) 等于0 等待同一进程组中任何子进程终止(但不包括组领导进程) -> 只算儿子
* (4) 大于0 仅等待指定进程ID的子进程终止
* wstatus: 整数指针,子进程返回的状态码会保存到该int
* options: 选项的值是以下常量之一或多个的按位或(OR)运算的结果;二进制对应选项,可多选:
* (1) WNOHANG 如果没有子进程终止,也立即返回;用于查看子进程状态而非等待
* (2) WUNTRACED 收到子进程处于收到信号停止的状态,也返回。
* (3) WCONTINUED(自Linux 2.6.10起)如果通过发送SIGCONT信号恢复了一个已停止的子进程,则也返回。
* return: (1) 成功等到子进程停止 返回pid
* (2) 没等到并且没有设置WNOHANG 一直等
* (3) 没等到设置WNOHANG 返回0
* (4) 出错返回-1
*/
pid_t waitpid(pid_t pid, int *wstatus, int options);
/*
更加全面的子进程监控和状态报告
*/
int waitid(idtype_t idtype, id_t id, siginfo_t *infop, int options);
示例:
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/wait.h>
int main(int argc, char const* argv[])
{
int state;//子进程的状态
printf("当前父进程%d\n", getpid());
__pid_t pid = fork();
// __pid_t son_pid = fork();
if (pid < 0) {
perror("fork");
return -1;
}
else if (pid == 0) {
char* args[] = { "/usr/bin/ping","-c" ,"50","baidu.com",NULL };
char* envs[] = { NULL };
printf("子进程%d,执行ping\n",getpid());
int re = execve(args[0], args, envs);
if (re < 0) {
perror("execve");
return -1;
}
}
else {
printf("父进程%d等待子进程%d\n", getpid(), pid);
waitpid(pid, &state, 0);
}
printf("执行完成\n");
return 0;
}
进程树
实质上,1号进程就是systemd,它由内核创建,是第一个进程,负责初始化系统,启动其他所有用户空间的服务和进程。它是所有进程的祖先。
在ps -ef的输出结果中,我们发现,CMD部分有的行带有[],而有的没有,前者属于内核线程,内核线程在内核空间执行,不占用任何用户空间资源,它们在技术上是线程,而在许多方面表现得像独立的进程,因此也会被ps命令检索到。第一个内核线程的pid为2,它是所有其它内核线程的祖先。
- pstree -p查看进程树
孤儿进程
孤儿进程(Orphan Process)是指父进程已结束或终止,而它仍在运行的进程。
当父进程结束之前没有等待子进程结束,且父进程先于子进程结束时,那么子进程就会变成孤儿进程。
我们可以得出结论:孤儿进程会被其祖先自动领养。此时的子进程因为和终端切断了联系,所以很难再进行标准输入使其停止了,所以写代码的时候一定要注意避免出现孤儿进程。