Linux进程退出
1. 进程退出的场景
- 代码运行完毕正常退出,结果正确
- 代码运行完毕正常退出,结果不正确
- 异常退出
2 .进程常见退出方式
- 正常退出
从main() 函数中返回return退出
调用exit()函数退出
调用_exit()函数退出 - 异常退出
由信号终止
return
return是常见的进程退出方式,执行return等同于执行exit函数,main()函数中return n函数返回值作为exit(n)函数的参数
exit()
exit()是c库函数其执行流程
首先调用用户定义的清理函数ateixt或者on_exit,清理标准输出缓存去(将输出缓存区内容写入文件),关闭所有打开的文件流等,然后回调用_eixt()函数。
需要注意的是由fork()函数创建的子进程分支里,正常情况下使用函数exit()是不正确的,这是因为使用它会导致标准输入输出的缓冲区被清空两次,而且临时文件可能被意外删除
另外需要注意的是调用exit() 函数会结束掉整个进程组中的所有进程(线程)
_exit()
正常退出的方式最终都会调用_exit()函数。
_exit()函数原形
void _eixt(int status); 包含在unistd.h库中
其中参数status定义了进程的终止状态, 父进程可以通过wait() 获取。
status(后16位):
Linux进程等待
创建子进程后如果父进程不等待,子进程退出后会成为僵尸进程,直到父进程来获取退出信息后才会释放剩余资源,并且此时该进程无法被信号杀死,继续占用资源造成内存泄露。
进程等待方式:
wait()
函数原形:pid_t wait(int * status);
返回值:退出的子进程的pid,失败返回-1
参数:输出型参数,用于获取子进程退出状态码,不关心可以置为NULL
是一个阻塞式等待,必须等到有一个子进程退出后获取退出状态,释放资源才可以返回
waitpid()
函数原形:pid_t waitpid(pid_t pid, int * status, int options);
返回值:当正常退出时返回退出进程的pid, 当调用失败(没有子进程)返回-1 errno会被设置成相应的值表示错误原因,可以通过perror函数进行打印错误,当options被设置为WNOHANG时如果此时没有已退出的子进程或者需要等待的子进程没有退出,则返回0
参数:
pid: pid 可以设置成-1表示等待任一进程,也可以设置成子进程的pid 指定需要等待的子进程。设置成0表示等待和进程组id相同的进程,当设置为<0时等待pid和其绝对值的相同的子进程;
status: 和wait参数status相同,获取退出状态码,如果不关心可以置为NULL
options:选项参数,设置成0同wait()进行阻塞等待,设置为WNOHANG时不阻塞如果没有退出的进程或者需要等待的子进程将直接返回0,另外这个参数还可以设置成其它属性。
另外两个函数可以解读状态码status
WTERMSIG(status)查看进程是否正常退出,若子进程正常退出则为真
WIFSIGNALED(status)查看进程退出状态码
也可以自己通过位运算来获取退出状态
1 #include <unistd.h>
2 #include <stdlib.h>
3 #include <errno.h>
4 #include <string.h>
5 #include <stdio.h>
6
7 int main()
8 {
9 pid_t pid = fork();
10 if(pid < 0)
11 {
12 perror("creat child error");
13 }
14 else if(pid == 0)
15 {
16 printf("this is child:%d\n", getpid());
17 sleep(10);
18 _exit(10);
19 }else if(pid > 0)
20 {
21 printf("this is parent:%d\n", getpid());
22 int sta = 0;
23 int ret = wait(&sta);
24 if(ret > 0 && (sta & 0xFF) == 0)//后8位为0时正常退出
25 {
26 printf("child exit code:%d\n", (sta >> 8)&0xFF);
27 }else if(ret > 0)//异常退出
28 {
29 printf("sig code:%d\n", sta & 0x7F);
30 }
31 }
32 return 0;
33 }