1、exit函数和return函数的主要区别是:

  • return:表示函数返回,当main函数return时表示程序运行结束.
  • exit():无论在哪个函数调用,都会导致进程结束,而进程结束,OS会回收所有线程。

    1)exit用于在程序运行的过程中随时结束程序,exit的参数是返回给父进程的。main函数结束后会返回到启动例程(_start函数)处,然后执行exit函数。exit函数运行时首先会执行由atexit()函数登记的函数,然后会做一些自身的清理工作,同时刷新所有输出流、关闭所有打开的流并且关闭通过标准I/O函数tmpfile()创建的临时文件。exit是结束一个进程,它将删除进程使用的内存空间,同时把错误信息返回父进程;而return是返回函数值并退出函数。通常情况:exit(0)表示程序正常, exit(1)和exit(-1)表示程序异常退出,exit(2)表示表示系统找不到指定的文件。在整个程序中,只要调用exit就结束(当前进程或者在main时候为整个程序)。
    2)return是语言级别的,它表示了调用堆栈的返回;return( )是当前函数返回,当然如果是在主函数main, 自然也就结束当前进程了,如果不是,那就是退回上一层调用。在多个进程时。如果有时要检测上个进程是否正常退出。就要用到上个进程的返回值,依次类推。而exit是系统调用级别的,它表示了一个进程的结束
    3)exit函数是退出应用程序,并将应用程序的一个状态返回给父进程,这个状态标识了应用程序的一些运行信息。
    4)和机器和操作系统有关的一般是: 0为正常退出,非0为非正常退出;

    2、进程环境与进程控制

    exit(int n)其实就是直接退出程序,因为默认的标准程序入口为 int main(int argc, char** argv),返回值是int型的。一般在shell下面,运行一个程序,然后使用命令echo $?就能得到该程序的返回值,也就是退出值,在main()里面,你可以用return n,也能够直接用exit(n)来做。unix默认的正确退出是返回0,错误返回非0。理论上exit可以返回小于256的任何整数。返回的不同数值主要是给调用者作不同处理的。
    单独的进程是返回给操作系统的。如果是多进程,是返回给父进程的。父进程里面调用waitpid()等函数得到子进程退出的状态,以便作不同处理。根据相应的返回值来让调用者作出相应的处理。总的说来,exit()就是当前进程把控制权返回给调用该程序的程序,括号里的是返回值,告诉调用程序该程序的运行状态。

    1)进程的开始:

    C程序是从main函数开始执行,原型如下:int main(int argc, char *argv[]);通常main的返回值是int型,正确返回0。如果main的返回值为void,某些编译器会给出警告,此时main的返回值通常是0。

    2)进程终止:

    C程序的终止分为两种:正常终止和异常终止。
    正常终止分为:return,exit,_exit,_Exit,pthreade_exit。异常中指分为:abort,SIGNAL,线程响应取消。
    主要说一下正常终止的前4种:即exit系列函数.

  1. #include <stdlib.h>
  2. void exit(int status);
  3. void _Exit(int status);
  4. #include <unistd.h>
  5. void _exit(int status);
  1. 以上3个函数的区别是:exit()(或return 0)会调用**终止处理程序**和用户空间的**标准I/O清理程序**(如fclose),_exit_Exit不调用而直接由内核接管进行清理,即直接调用sys_exit syscall。因此,在main函数中exit(0)等价于return 0

3)atexit终止处理程序:

函数名: atexit
头文件:#include
功 能: 注册终止函数(即main执行结束后调用的函数)
用 法: void atexit(void (*func)(void));
注 意:ISO C规定,一个进程最多可登记32个终止处理函数。exit调用这些注册函数的顺序与它们登记时候的顺序相反。同一个函数如若登记多次,则也会被调用多次。
原型如下:

  1. #include <stdlib.h>
  2. int atexit(void (*func)(void));


其中参数是一个函数指针,指向终止处理函数,该函数无参无返回值。atexit函数本身成功调用后返回0。
以下面的程序为例:

  1. #include <stdlib.h>
  2. static void myexit1()
  3. {
  4. printf("first exit handler\n");
  5. }
  6. static void myexit2()
  7. {
  8. printf("second exit handler\n");
  9. }
  10. int main()
  11. {
  12. atexit(my_exit2);
  13. atexit(my_exit1);
  14. atexit(my_exit1);
  15. printf("main is done\n");
  16. return 0;
  17. }
  1. 运行结果:<br /> $ ./a.out<br /> main is done<br /> first exit handler<br /> first exit handler<br /> second exit handler
  2. 注意上面的结果,可以发现这些函数由exit按登记相反的顺序自动调用(先myexit1myexit2)。如果同一函数登记多次, 也会被调用多次(如这里的myexit1)。程序退出的时候利用atexit函数调用了这些处理函数。但是如果用_exit()退出程序,则它不关闭任何文件,不清除任何缓冲器、也不调用任何终止函数!

3.exit和_exit的区别

113.png
c语言库函数的IO函数族(我们熟知的printf 、fopen 、fread 、fwrite)是buffer io,也就是打开文件时会分配一片内存作为缓冲区(这个缓冲区属于用户空间)。读出或者写入的数据优先送入缓冲区中。等到一定条件才写入文件。内核提供的系统函数是unbuffered io,注意是没用户空间的缓冲区,但是有在内核空间的缓冲区。
exit是个库函数,exit退出进程之前,会执行一些清除动作,它会先执行一些清除操作,例如调用执行各终止处理函数、关闭所有标准IO并清理其缓存等,然后调用sys_exit。清理IO缓存即检查进程打开的文件,将文件缓冲区的内容写入到磁盘中。最后再调用sysexit系统调用。
下面我就举个例子说明下吧,

  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #include <unistd.h>
  4. int main()
  5. {
  6. int ret = 0;
  7. printf("I'm ok!\n");
  8. printf("Good good study!");
  9. _exit(0);
  10. }
  11. //执行输出:
  12. //I'm ok!
  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #include <unistd.h>
  4. int main()
  5. {
  6. int ret = 0;
  7. printf("I'm ok!\n");
  8. printf("Good good study!");
  9. exit(0);
  10. }
  11. //执行输出:
  12. //I'm ok!
  13. //Good good study!