Linux中的进程等待
为什么需要进程等待
我们都知道子进程退出后(kill掉),父进程还在时,子进程会变成僵尸进程,会造成内存泄漏,而进程等待,就是用来等待父进程回收清理子进程的空间,和查看子进程完成任务的情况(就是查看退出码)的过程。
进程等待方法
父进程可以通过两个函数来获得子进程退出情况,这两个函数分别是wait()和waitpid()。
wait()函数
int status = 0;
int id = wait(&status);
子进程正常退出时wait()的返回值返回子进程的pid,wait()后status使用了整形前16位,正常退出时,status的16位的前8位都是零,后8位者表示的是子进程的退出码,看如下代码操作:
1 #include <stdio.h>
2 #include <unistd.h>
3 #include <stdlib.h>
4 #include <sys/wait.h>
5
6 int main()
7 {
8 pid_t id = fork();
9 if(id == 0)
10 {
11 printf("子进程运行,pid:%d\n",getpid());
12 exit(1);
13 }
14 else
15 {
16 int st;
17 int ret = wait(&st);
18 printf("pid:%d,returnNum:%d\n",ret,(st >> 8)&(0x7f));
19 }
20
21 return 0;
22 }
上面的代码运行结果如下:
第一行表示子进程运行,且pid=16550,第二行是父进程运行的结果,显示的是wait()函数的返回值,返回值是和子进程pid一样的,且returnNum=1,说明子进程正常退出。
waitpid()函数
int status = 0;
int id = waitpid(pid,&status,0);
waitpid()函数有三个输入参数,第一个参数输入-1时,表示任意一个子进程退出时的等待,如果输入子进程pid,则函数查看的是指定pid的子进程的退出情况;第二个参数和wait()函数一样;第三个参数输入0时,表示阻塞等待,输入WNOHANG,表示非阻塞等待。而waitpid()的返回值,如果子进程正常退出,waitpid()的返回值时子进程的pid。
阻塞等待与非阻塞等待
进程的阻塞等待和非阻塞等待,阻塞等待就是父进程等待子进程运行完了,父进程再运行,非阻塞等待就是父子进程同时运行。看如下代码:
1 #include <stdio.h>
2 #include <unistd.h>
3 #include <stdlib.h>
4 #include <sys/wait.h>
5
6 int main()
7 {
8 pid_t id = fork();
9 if(id == 0)
10 {
11 printf("子进程运行,pid:%d\n",getpid());
12 exit(1);
13 }
14 else
15 {
16 int st;
17 int ret = waitpid(-1,&st,0);//阻塞等待
18 printf("ret:%d\n",ret);
19 if( WIFEXITED(st) && ret == id)
20 {
21 printf("pid:%d,returnNum:%d\n",ret,WEXITSTATUS(st));
22 }
23 }
24 return 0;
25 }
WIFEXITED(st)这个函数,如果子进程正常退出,返回值为真,反之为假;WEXITSTATUS(st)这个函数当WIFEXITED(st)为真时,返回子进程pid,运行结果如下: