[Linux]进程控制
文章目录
进程退出情况分类
- 进程正常执行完成 - 运行结果正确- 运行结果错误
- 进程异常终止 – (进程产生错误后,收到了操作系统的信号)
进程退出码的理解
进程主体功能执行完毕后会(return)返回退出码,进程正常执行完成后,根据运行结果的不同会返回不同的退出码,不同的退出码代表不同的含义,其中0退出码代表运行结果正确,其他数字分别代表一种运行错误结果。通过查验退出码可以得知,进程运行情况。
查看C语言的退出码含义
C语言库函数遵循C语言的退出码标准,使用C语言的库函数
strerror
函数可以将C语言标准中的退出码转换成说明其情况的字符串,编写如下代码查看:
#include<stdio.h>#include<string.h>intmain(){int i =0;for(i =0; i <150; i++)//150并非准确的退出码个数,为预估值{printf("%d -> %s\n", i,strerror(i));}return0;}
编译代码执行程序查看结果:
查看进程退出码
Linux系统中
echo $?
可以查看最近一次退出的进程的退出码:
ls进程遵守C语言标准的返回码,因此在文件不存在时返回码是2。
进程退出方式
- 调用_exit函数
_exit函数是Linux操作系统提供的系统接口,用户可以调用该系统调用接口让操作系统退出进程,该函数所处的头文件及参数列表如下:
_exit的参数会作为进程退出的退出码,编写如下代码测试:
#include<stdio.h>#include<unistd.h>intmain(){_exit(123);return0;}
编译代码执行程序查看结果:
说明: _exit函数无论在任何位置调用都能退出进程。
- 调用exit函数
exit函数是C语言提供的库函数,用户可以调用该库函数让操作系统退出进程,该函数所处的头文件及参数列表如下:
exit的参数会作为进程退出的退出码,编写如下代码测试:
#include<stdio.h>#include<unistd.h>intmain(){exit(222);return0;}
编译代码执行程序查看结果:
说明: exit函数无论在任何位置调用都能退出进程。
_exit函数和exit函数的区别
exit函数是C语言的库函数,库函数的实现要依赖于开发环境的具体实现,由于只有操作系统有退出进程(释放pcb+代码和数据)的权利,因此exit函数是封装系统接口_exit函数实现的,此外exit函数还会执行用户定义的清理函数,冲刷缓冲区,关闭流等操作。如下图:
为了验证exit函数和_exit函数的区别,首先编写如下代码:
#include<stdio.h>#include<unistd.h>intmain(){printf("hello world");sleep(2);_exit(123);return0;}
编译代码执行程序查看结果:
然后编写如下代码:
#include<stdio.h>#include<unistd.h>#include<stdlib.h>intmain(){printf("hello world");sleep(2);exit(222);return0;}
编译代码执行程序查看结果:
可以看到调用_exit的进程由于没有冲刷缓冲区,导致最终输出没有打印到屏幕上,而调用exit的进程冲刷了缓冲区,使得输出打印到了屏幕上。
- main函数中调用return
return是一种更常见的退出进程方法。在main函数中执行return(n)等同于执行exit(n),因为调用main的运行时函数会将main的返回值当做exit的参数。
进程等待
进程等待是调用系统接口,释放僵尸状态的子进程所占的内存资源,并且接收进程的退出码和退出信号。
wait函数
wait函数会一直等待直到子进程退出进入僵尸状态,然后将子进程回收。该函数的头文件和参数列表如下:
- 参数列表为NULL的功能只有回收自己成所占的内存资源。
- 参数为输出型参数,接收子进程的退出码和退出信号。
- 等待成功返回等待到的子进程的id,等待失败返回-1。
为了验证wait函数的作用,编写如下代码:
#include<stdio.h>#include<unistd.h>#include<sys/types.h>#include<sys/wait.h>intmain(){pid_t id =fork();if(id ==0){//子进程int cnt =5;while(1){printf("我是子进程,我还能活%dS,我的pid是:%d, 我的ppid是:%d\n", cnt--,getpid(),getppid());sleep(1);if(cnt ==0)exit(123);}}//父进程sleep(10);pid_t ret_id =wait(NULL);printf("我是父进程,我等待子进程成功,我的pid:%d, 我的ppid:%d, ret_id:%d\n",getpid(),getppid(), ret_id);sleep(3);return0;}
编译代码执行程序查看结果:
在子进程执行5秒后退出进入了僵尸状态,再然后父进程调用了wait函数回收了子进程。
waitpid函数
wait函数会等待子进程退出进入僵尸状态,然后将子进程回收。该函数的头文件和参数列表如下:
- 参数pid- Pid=-1,等待任一个子进程。与wait等效。- Pid>0.等待其进程ID与pid相等的子进程。
- 参数status- 输出型参数,接收等待进程的退出码和退出信号。
- 参数options- WNOHANG: 若pid指定的子进程没有结束,则waitpid()函数返回0,不予以等待。若正常结束,则返回该子进程的ID。- 0:若pid指定的子进程没有结束,一直等待直到指定的子进程结束。
- 返回值- 当正常返回的时候waitpid返回收集到的子进程的进程ID;- 如果设置了选项WNOHANG,而调用中waitpid发现没有已退出的子进程可收集,则返回0;- 如果调用中出错,则返回-1,这时errno会被设置成相应的值以指示错误所在;
参数status的结构示意图如下:
status中的次低8位(倒数第9位到倒数第16位)二进制数转化成十进制就是进程退出码,status中低7位二进制数转换成十进制就是进程退出信号。(wait函数接收的status参数结构相同。)
为了验证waitpid函数的作用,编写如下代码:
#include<stdio.h>#include<unistd.h>#include<sys/types.h>#include<sys/wait.h>intmain(){pid_t id =fork();if(id ==0){//子进程int cnt =5;while(1){printf("我是子进程,我还能活%dS,我的pid是:%d, 我的ppid是:%d\n", cnt--,getpid(),getppid());sleep(1);if(cnt ==0)exit(123);}}//父进程int status =0;waitpid(id,&status,0);printf("我是父进程,我等待子进程成功,我的pid:%d, 我的ppid:%d, ret_code:%d, ret_signal:%d\n",getpid(),getppid(),(status>>8)&0xFF, status&0x7F);return0;}
编译代码执行程序查看结果:
可以看出waitpid函数确实接收到了子进程的退出码123,由于是正常执行完的退出信号为0。
wait/waitpid获取退出码和退出信号的原理
在子进程的pcb中会存在存储进程退出码和退出信号的字段,子进程退出后进入僵尸状态时,pbc还保留在内存中,wait/waitpid函数只需要找到对应pid的pcb就可以获得到子进程的退出码和退出信号。
waitpid的阻塞等待和非阻塞等待
waitpid第三个参数设置为0,表示waipid采用阻塞等待的方式,也就是调用waitpid后,进程会一直等待子进程的退出,如果子进程不退出,调用waitpid的进程就会阻塞,直到子进程退出。
waitpid第三个参数设置为WNOHANG,表示waipid采用非阻塞等待的方式,也就是调用waitpid后,立刻查看进程是否退出,如果没有退出就什么都不做并返回0,如果进程退出了,就获取退出码和退出信号并释放内存空间。为了验证waitpid函数的非阻塞等待作用,编写如下代码:
#include<stdio.h>#include<string.h>#include<unistd.h>#include<stdlib.h>#include<sys/types.h>#include<sys/wait.h>intmain(){pid_t id =fork();if(id ==0){//子进程int cnt =5;while(1){printf("我是子进程,我还能活%dS,我的pid是:%d, 我的ppid是:%d\n", cnt--,getpid(),getppid());sleep(1);if(cnt ==0)exit(123);}}int status =0;while(1){pid_t ret_id =waitpid(id,&status, WNOHANG);if(ret_id <0){perror("子进程出错\n");exit(1);}elseif(ret_id ==0){printf("子进程还没退出,我再等等\n");sleep(1);continue;}else{printf("我是父进程,我等待子进程成功,我的pid:%d, 我的ppid:%d, ret_code:%d, ret_signal:%d\n",getpid(),getppid(),(status>>8)&0xFF, status&0x7F);exit(0);}}return0;}
编译代码执行程序查看结果:
使用宏获取进程退出码
- WIFEXITED(status): 若为正常终止子进程返回的状态,则为真。(查看进程是否是正常退出)
- WEXITSTATUS(status): 若WIFEXITED非零,提取子进程退出码。(查看进程的退出码)
status为wait/waitpid函数的参数。
编写如下代码测试宏获取进程退出码:
#include<stdio.h>#include<string.h>#include<unistd.h>#include<stdlib.h>#include<sys/types.h>#include<sys/wait.h>intmain(){pid_t id =fork();if(id ==0){//子进程int cnt =5;while(1){printf("我是子进程,我还能活%dS,我的pid是:%d, 我的ppid是:%d\n", cnt--,getpid(),getppid());sleep(1);if(cnt ==0)exit(123);}}int status =0;while(1){pid_t ret_id =waitpid(id,&status, WNOHANG);if(ret_id <0){perror("子进程出错\n");exit(1);}elseif(ret_id ==0){printf("子进程还没退出,我再等等\n");sleep(1);continue;}else{if(WIFEXITED(status)){printf("进程正常退出,code:%d\n",WEXITSTATUS(status));exit(0);}else{printf("进程异常退出,signal:%d\n", status&0x7F);exit(0);}}}return0;}
编译代码执行程序查看结果:
版权归原作者 好想写博客 所有, 如有侵权,请联系我们删除。