exit()与_exit()函数的区别

进程就好比人一样有其生命,我们通过fork()函数来创建一个进程,那么我们又是如何来中止进程呢。

1.Linux中的进程退出

进程退出表示进程即将结束。在Linux中进程退出分为了正常退出和异常退出两种。

(1)正常退出

a. 在main()函数中执行return;

b.调用exit()函数;

c.调用_exit()函数。

(2)异常退出

a.调用about函数;

b.进程收到某个信号,而该信号使程序终止。

补充:不管是那种退出方式,系统最终都会执行内核中的同一代码。这段代码用来关闭进程所用已打开的文件描述符,释放它所占用的内存和其他资源。

比较以上几种退出方式的不同点

①exit和return 的区别:

a.exit是一个函数,有参数。exit执行完后把控制权交给系统;

b.return是函数执行完后的返回。renturn执行完后把控制权交给调用函数。

②exit和abort的区别:

a.exit是正常终止进程;

b.about是异常终止。

现在我们重点了解exit()和_exit()函数。

2.exit()和_exit()

exit()就是退出,传入的参数是程序退出时的状态码,0表示正常退出,其他表示非正常退出,一般都用-1或者1,标准C里有EXIT_SUCCESS和EXIT_FAILURE两个宏,用exit(EXIT_SUCCESS);可读性比较好一点。

作为系统调用而言,_exit和exit是一对孪生兄弟,它们究竟相似到什么程度,我们可以从Linux的源码中找到答案:


#define __NR__exit __NR_exit /* 摘自文件include/asm-i386/unistd.h第334行 */

"__NR_"是在Linux的源码中为每个系统调用加上的前缀,请注意第一个exit前有2条下划线,第二个exit前只有1条下划线。

这时随便一个懂得C语言并且头脑清醒的人都会说,_exit和exit没有任何区别,但我们还要讲一下这两者之间的区别,这种区别主要体现在它们在函数库中的定义。

_exit在Linux函数库中的原型是:


#include “unistd.h”

void _exit(int status);

和exit比较一下,exit()函数定义在stdlib.h中,而_exit()定义在unistd.h中,从名字上看,stdlib.h似乎比 unistd.h高级一点,那么,它们之间到底有什么区别呢?

_exit()函数的作用最为简单:直接使进程停止运行,清除其使用的内存空间,并销毁其在内核中的各种数据结构;exit() 函数则在这些基础上作了一些包装,在执行退出之前加了若干道工序,也是因为这个原因,有些人认为exit已经不能算是纯粹的系统调用。

exit()函数与_exit()函数最大的区别就在于exit()函数在调用exit系统调用之前要检查文件的打开情况,把文件缓冲区中的内容写回文件,就是"清理I/O缓冲"。

=========================================================

(1)exit和_exit函数都是用来终止进程的。

当程序执行到exit或_exit时,系统无条件的停止剩下所有操作,清除包括PCB(进程控制块)在内的各种数据结构,并终止本进程的运行。

exit()在结束调用它的进程之前,要进行如下步骤:

①调用atexit()注册的函数(出口函数);按ATEXIT注册时相反的顺序调用所有由它注册的函数,这使得我们可以指定在程序终止时执行自己的清理动作.例如,保存程序状态信息于某个文件,解开对共享数据库上的锁等.

②cleanup();关闭所有打开的流,这将导致写所有被缓冲的输出,删除用TMPFILE函数建立的所有临时文件.

③最后调用_exit()函数终止进程。

_exit做3件事(man):

①Any  open file descriptorsbelonging to the process are closed

②any children of the process are inherited  by process 1, init

③the process‘s parent is sent a SIGCHLD signal

exit执行完清理工作后就调用_exit来终止进程。

此外,另外一种解释:

简单的说,exit函数将终止调用进程,在退出程序之前,所有文件关闭,缓冲输出内容将刷新定义,并调用所有已刷新的“出口函数”(由atexit定义)。_exit函数是由Posix定义的,不会运行exithandler和signal
handler,在UNIX系统中不会flush标准I/O流。简单的说,_exit终止调用进程,但不关闭文件,不清除输出缓存,也不调用出口函数。

共同:不管进程是如何终止的,内核都会关闭进程打开的所有file descriptors,释放进程使用的memory!

=========================================================

(2)exit在头文件stdlib.h中声明,而_exit()声明在头文件unistd.h中声明。 exit中的参数exit_code为0代表进程正常终止,若为其他值表示程序执行过程中有错误发生。

=========================================================

(3)exit()和_exit()的区别:

a._exit()执行后立即返回给内核,而exit()要先执行一些清除操作,然后将控制权交给内核。

b.调用_exit函数时,其会关闭进程所有的文件描述符,清理内存以及其他一些内核清理函数,但不会刷新流(stdin, stdout, stderr ...); exit函数是在_exit函数之上的一个封装,其会调用_exit,并在调用之前先刷新流。

补充:

exit()函数在调用exit系统之前要检查文件的打开情况,把文件缓冲区的内容写回文件。由于Linux的标准函数库中,有一内存中都有一片缓冲区。每次读文件时,会连续的读出若干条记录,这样在下次读文件时就可以直接从内存的缓冲区读取;同样,每次写文件的时候也仅仅是写入内存的缓冲区,等满足了一定的条件(如达到了一定数量或遇到特定字符等),再将缓冲区中的内容一次性写入文件。这种技术大大增加了文件读写的速度,但也给编程代来了一点儿麻烦。比如有一些数据,认为已经写入了文件,实际上因为没有满足特定的条件,它们还只是保存在缓冲区内,这时用_exit()函数直接将进程关闭,缓冲区的数据就会丢失。因此,要想保证数据的完整性,就一定要使用exit()函数。

=========================================================

通过实例来看看它们之间的区别:

函数实例1 : exit.c

1 #include<stdio.h>

2 #include<stdlib.h>

3

4 int main()

5 {

6     printf("usingexit----\n");

7     printf("This is thecontent in
buffer\n");

8     exit(0);

9 }

函数经过调试后

[email protected]:~/work/process_thread/exit$gcc exit.c -o exit

[email protected]:~/work/process_thread/exit$./exit

执行结果为:

using exit----

This is the content in buffer

=========================================================

函数实例2:_exit1.c

1 #include<stdio.h>

2 #include<unistd.h>

3

4 int main(void)

5 {

6     printf("using_exit--\n"); //如果此处不加“\n”的话,这条信息有可能也不会显示在终端上。

7     printf("This is thecontent in
buffer");

8     _exit(0);

9 }

函数经过调试后

[email protected]:~/work/process_thread/exit$gcc _exit1.c -o _exit1

[email protected]:~/work/process_thread/exit$./_exit1

执行结果为 :

using _exit--

在一个进程调用了exit之后,该进程并不会马上完全消失,而是留下一个称为僵尸进程(Zombie)的数据结构。僵尸进程是一种非常特殊的进程,它几乎已经放弃了所有的内存空间,没有任何可执行代码,也不能被调度,仅仅在进程列表中保留一个位置,记载该进程的退出状态等信息供其它进程收集,除此之外,僵尸进程不再占有任何内存空间。

=========================================================

函数实例2:_exit2.c

1 #include<stdio.h>

2 #include<unistd.h>

3 int main()

4 {

5     printf("%c", ‘c‘);

6     _exit(0);

7 }

函数经过调试后

[email protected]:~/work/process_thread/exit$gcc _exit2.c
-o _exit2

[email protected]:~/work/process_thread/exit$./_exit2

程序并没有输出"c", 说明_exit()没有进行io flush。

=========================================================

1.printf函数就是使用缓冲I/O的方式,该函数在遇到“\n”换行符时自动的从缓冲区中将记录读出。所以exit()将缓冲区的数据写完后才退出,而_exit()函数直接退出。

2.大家也可以把函数实例2中的printf("Thisis
the content in buffer");改为printf("This isthe
content in buffer\n")(即在printf中最后加一个\n看运行结果是什么,为什么会产生这样的结果呢?)

补充:

父子进程终止的先后顺序不同会产生不同的结果。

(1)父进程先于子进程终止

此种情况就是我们前面所用的孤儿进程。当父进程先退出时,系统会让init进程接管子进程 。

(2)子进程先于父进程终止,而父进程又没有调用wait函数

此种情况子进程进入僵死状态,并且会一直保持下去直到系统重启。子进程处于僵死状态时,内核只保存进程的一些必要信息以备父进程所需。此时子进程始终占有着资源,同时也减少了系统可以创建的最大进程数。

补充:一个已经终止、但是其父进程尚未对其进行善后处理(获取终止子进程的有关信息,释放它仍占有的资源)的进程被称为僵死进程(zombie)。ps命令将僵死进程的状态打印为Z。

(3)子进程先于父进程终止,而父进程调用了wait函数

此时父进程会等待子进程结束。

整理来自:

http://blog.chinaunix.net/uid-12461657-id-3140887.html

http://www.cnblogs.com/hnrainll/archive/2011/08/17/2142001.html

时间: 2024-10-03 11:27:00

exit()与_exit()函数的区别的相关文章

exit()与_exit()函数的区别(Linux系统中)

注:exit()就是退出,传入的参数是程序退出时的状态码,0表示正常退出,其他表示非正常退出,一般都用-1或者1,标准C里有EXIT_SUCCESS和EXIT_FAILURE两个宏,用exit(EXIT_SUCCESS);可读性比较好一点. 作为系统调用而言,_exit和exit是一对孪生兄弟,它们究竟相似到什么程度,我们可以从Linux的源码中找到答案: #define __NR__exit __NR_exit /* 摘自文件include/asm-i386/unistd.h第334行 */

exit()和_exit()函数(转)

转自    http://www.cnblogs.com/hnrainll/archive/2011/08/17/2142001.html进程退出 1.在Linux中任何让一个进程退出 进程退出表示进程即将结束.在Linux中进程退出分为了正常退出和异常退出两种. 1>正常退出  a. 在main()函数中执行return . b.调用exit()函数 c.调用_exit()函数 2>异常退出  a.调用abort函数 b.进程收到某个信号,而该信号使程序终止. Tiger-John说明:不管

exit()和_exit()函数

具体内容参考:http://www.cnblogs.com/hnrainll/archive/2011/08/17/2142001.html 项目中用到的地方: if (state[state.articleName] && state[state.articleName].exit) { state[state.articleName].exit()} 如参考中提到的exit和return的区别:1.a.exit是一个函数,有参数.exit执行完后把控制权交给系统2.b.return是函

Linux多任务编程之五:exit()和_exit()函数(转)

来源:CSDN  作者:王文松   转自:Linux公社 ------------------------------------------------------------------------------------------------------------------------------------------------ wait()和waitpid() 函数说明 wait()函数用于使父进程(也就是调用wait()的进程)阻塞,直到一个子进程结束或者该进程接收到了一个指

回收进程用户空间资源 exit()函数 _exit()函数 atexit()函数 on_exit()函数

摘要:本文主要讲述进程的终止方式,以及如何使用exit()函数来终止进程,回收进程用户空间资源:分析了exit()函数与_exit()函数,return关键字的差异.同时详细解读了如何使用atexit()和on_exit()函数来注册终止处理程序. 进程终止.回收资源 1.进程终止方式 在内核中,程序执行的唯一方法是调用一个exec函数.而进程自愿终止的唯一方法是显示或隐式地调用_exit()或_Exit(). 进程有5种正常终止方式: (1)常见的一种是,在main函数中执行return语句,

Linux中exit与_exit的区别

在exit,_exit的区别 - exit()与_exit()函数的区别(Linux系统中)2012-03-20 15:19:53 分类: LINUX 注:exit()就是退出,传入的参数是程序退出时的状态码,0表示正常退出,其他表示非正常退出,一般都用-1或者1,标准C里有EXIT_SUCCESS和EXIT_FAILURE两个宏,用exit(EXIT_SUCCESS);可读性比较好一点. 作为系统调用而言,_exit和exit是一对孪生兄弟,它们究竟相似到什么程度,我们可以从Linux的源码中

程序代码中退出函数exit()与返回函数return ()的区别

  exit(0):正常运行程序并退出程序:   exit(1):非正常运行导致退出程序:   return():返回函数,若在主函数中,则会退出函数并返回一值.  解析: 1. return返回函数值,是关键字: exit 是一个函数. 2. return是语言级别的,它表示了调用堆栈的返回:而exit是系统调用级别的,它表示了一个进程的结束. 3. return是函数的退出(返回):exit是进程的退出. 4. return是C语言提供的,exit是操作系统提供的(或者函数库中给出的).ex

Linux C _exit函数与exit函数的联系与区别

一.联系 1.功能上,_exit和exit函数都是让进程正常退出,即关闭进程所打开的文件描述符,释放已占用内存和其他资源. 二.区别 1._exit函数在头文件unistd.h中声明,而exit在头文件stdlib.h中声明. 2.执行_exit函数后,控制权立即返回给内核,而exit函数要先执行一些清除操作,然后才将控制权交给内核. 3._exit函数不会刷新I/O缓冲区,可能造成数据丢失,而exit函数是在_exit函数之上的一个封装,它在调用_exit函数之前,会先刷新I/O缓冲区,保证了

linux exit 和 _exit的区别

今天仔细看了一下exit和_exit这两个函数的区别,实际上exit也是调用了_exit退出函数的,只不过在调用_exit之前,exit还进行了一些多余的工作,也正是因为这样,相比起来exit就没有那么接近底层的系统调用,更应该说是包装过的标准C库函数._exit包含在头文件unistd.h中,exit包含在头文件stdlib.h中,我们来查看一下他们的函数原型. man _exit : 很显然,_exit函数做这三件事情: 1. 让调用的进程马上终止. 2.关闭所有由这个进程打开的文件描述符.