_exit(0)与exit(0)
-
exit ():调用exit函数之后,它首先会执行一系列的清理处理,包括调用执行各终止处理程序,关闭所有标准IO流等,然后进入内核。
-
_exit ():与exit不同的是,它不进行清理工作而直接进入内核。此函数由POSIX.1说明,放在unistd.h里面。
-
return :函数返回传递参数的,当然在main函数里就是结束进程了。也就是说,在main()里面,你可以用return n,也能够直接用exit(n)来做。
验证代码如下:
#include <stdlib.h>
#include<unistd.h>
#include <stdio.h>
int main()
{
printf("hello world!");
_exit(0);
//exit(0);
}
结果如下:
细心的读者可以发现:
此时终端并没有打印hello world!
而重定位输出到文件后,依旧没有查看到打印信息。
而我们不妨将测试代码中的_exit(0)改成exit(0)试试
结果如下:
你会发现终端打印出信息,同时重定位也有信息。
_exit(0)与exit(0)解释如下:
由于printf函数是行缓冲的(因为它要往终端输出数据),而且要打印的字符串不带换行符,因此在它没有遇到换行符或者没有填满缓冲区之前不会进行实际的IO操作,而在_exit函数有立即进入内核没有处理IO缓冲区,所以我们在终端上看不到hello world语句。当我们调用exit()函数时,在终端看到了hello world。
return、_exit()与exit的区别:
头文件:
exit:#include<stdlib.h>
_exit:#include<unistd.h>
_exit()函数:直接使进程停止运行,清除其使用的内存空间,并销毁其在内核中的各种数据结构;
exit()函数则在这些基础上作了一些包装,在执行退出之前加了若干道工序。
exit()函数与_exit()函数最大的区别就在于 exit()函数在调用 exit 系统调用之前要检查文件的打开情况,把文件缓冲区中的内容写回文件。
exit是系统调用级别的,它表示了一个进程的结束,它将删除进程使用的内存空间,同时把错误信息返回父进程。通常情况:exit(0)表示程序正常, exit(1)和exit(-1)表示程序异常退出,exit(2)表示系统找不到指定的文件。在整个程序中,只要调用exit就结束。
return是语言级别的,它表示了调用堆栈的返回;return是返回函数值并退出函数,通常0为正常退出,非0为非正常退出
退出过程
1.调用atexit()注册的函数(出口函数);按ATEXIT注册时相反的顺序调用所有由它注册的函数,这使得我们可以指定在程序终止时执行自己的清理动作.例如,保存程序状态信息于某个文件,解开对共享数据库上的锁等.
2.cleanup();关闭所有打开的流,这将导致写所有被缓冲的输出,删除用TMPFILE函数建立的所有临时文件.
3.最后调用_exit()函数终止进程。
_exit做3件事(man):
1,Any open file descriptors belonging to the process are closed
2,any children of the process are inherited by process 1, init
3,the process’s parent is sent a SIGCHLD signal
exit执行完清理工作后就调用_exit来终止进程。
最后,我们配上一幅图加以记忆: