Linux系统命令中exit与exit的区别linux命令exit什么意思




Linux系统命令中exit与exit的区别linux命令exit什么意思

2022-07-20 20:29:12 网络知识 官方管理员

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

作为系统调用而言,_exitexit是一对孪生兄弟,它们究竟相似到什么程度,我们可以从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(intstatus);

和exit比较一下,exit()函数定义在stdlib.h中,而_exit()定义在unistd.h中,从名字上看,stdlib.h似乎比unistd.h高级一点,那么,它们之间到底有什么区别呢?_exit()函数的作用最为简单:直接使进程停止运行,清除其使用的内存空间,并销毁其在内核中的各种数据结构;exit()函数则在这些基础上作了一些包装,在执行退出之前加了若干道工序,也是因为这个原因,有些人认为exit已经不能算是纯粹的系统调用。exit()函数与_exit()函数最大的区别就在于exit()函数在调用exit系统调用之前要检查文件的打开情况,把文件缓冲区中的内容写回文件,就是"清理I/O缓冲"。

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

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

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

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

_exit3件事(man):1Anyopenfiledescriptorsbelongingtotheprocessareclosed2anychildrenoftheprocessareinheritedbyprocess1,init3theprocess‘sparentissentaSIGCHLDsignal

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

此外,另外一种解释:

简单的说,exit函数将终止调用进程。在退出程序之前,所有文件关闭,缓冲输出内容将刷新定义,并调用所有已刷新的出口函数(由atexit定义)。

_exit:该函数是由Posix定义的,不会运行exithandler和signalhandler,在UNIX系统中不会flush标准I/O流。

简单的说,_exit终止调用进程,但不关闭文件,不清除输出缓存,也不调用出口函数。

共同:

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

更详细的介绍:

Callingexit()Theexit()functioncausesnormalprogramtermination.

Theexit()functionperformsthefollowingfunctions:

1.AllfunctionsregisteredbytheStandardCatexit()functionarecalledinthereverseorderofregistration.Ifanyofthesefunctionscallsexit(),theresultsarenotportable.2.Allopenoutputstreamsareflushed(datawrittenout)andthestreamsareclosed.

3.Allfilescreatedbytmpfile()aredeleted.

4.The_exit()functioniscalled.Calling_exit()The_exit()functionperformsoperatingsystem-specificprogramterminationfunctions.Theseinclude:1.Allopenfiledescriptorsanddirectorystreamsareclosed.

2.Iftheparentprocessisexecutingawait()orwaitpid(),theparentwakesupandstatusismadeavailable.

3.Iftheparentisnotexecutingawait()orwaitpid(),thestatusissavedforreturntotheparentonasubsequentwait()orwaitpid().4.ChildrenoftheterminatedprocessareassignedanewparentprocessID.Note:theterminationofaparentdoesnotdirectlyterminateitschildren.5.IftheimplementationsupportstheSIGCHLDsignal,aSIGCHLDissenttotheparent.6.Severaljobcontrolsignalsaresent.

为何在一个fork的子进程分支中使用_exit函数而不使用exit函数?‘exit()’‘_exit()’有不少区别在使用‘fork()’,特别是‘vfork()’时变得很突出。

‘exit()’‘_exit()’的基本区别在于前一个调用实施与调用库里用户状态结构(user-modeconstructs)有关的清除工作(clean-up),而且调用用户自定义的清除程序(自定义清除程序由atexit函数定义,可定义多次,并以倒序执行),相对应,_exit函数只为进程实施内核清除工作。在由‘fork()’创建的子进程分支里,正常情况下使用‘exit()’是不正确的,这是因为使用它会导致标准输入输出(stdio:StandardInputOutput)的缓冲区被清空两次,而且临时文件被出乎意料的删除(临时文件由tmpfile函数创建在系统临时目录下,文件名由系统随机生成)。在C++程序中情况会更糟,因为静态目标(staticobjects)的析构函数(destructors)可以被错误地执行。(还有一些特殊情况,比如守护程序,它们的父进程需要调用‘_exit()’而不是子进程;适用于绝大多数情况的基本规则是,‘exit()’在每一次进入‘main’函数后只调用一次。)在由‘vfork()’创建的子进程分支里,‘exit()’的使用将更加危险,因为它将影响父进程的状态。

#include<sys/types.h>;#include<stdio.h>intglob=6;/*externalvariableininitializeddata*/intmain(void){intvar;/*automaticvariableonthestack*/pid_tpid;var=88;printf("beforevfork\n";/*wedon‘tflushstdio*/if((pid=vfork())<0)printf("vforkerror\n";elseif(pid==0){/*child*/glob++;/*modifyparent‘svariables*/var++;exit(0);/*childterminates*///子进程中最好还是用_exit(0)比较安全。}/*parent*/printf("pid=%d,glob=%d,var=%d\n",getpid(),glob,var);exit(0);}在Linux系统上运行,父进程printf的内容输出:pid=29650,glob=7,var=89

子进程关闭的是自己的,虽然他们共享标准输入、标准输出、标准出错等打开的文件,子进程exit时,也不过是递减一个引用计数,不可能关闭父进程的,所以父进程还是有输出的。

但在其它UNIX系统上,父进程可能没有输出,原因是子进程调用了exit,它刷新关闭了所有标准I/O流,这包括标准输出。虽然这是由子进程执行的,但却是在父进程的地址空间中进行的,所以所有受到影响的标准I/OFILE对象都是在父进程中的。当父进程调用printf时,标准输出已被关闭了,于是printf返回-1

Linux的标准函数库中,有一套称作"高级I/O"的函数,我们熟知的printf()fopen()fread()fwrite()都在此列,它们也被称作"缓冲I/ObufferedI/O",其特征是对应每一个打开的文件,在内存中都有一片缓冲区,每次读文件时,会多读出若干条记录,这样下次读文件时就可以直接从内存的缓冲区中读取,每次写文件的时候,也仅仅是写入内存中的缓冲区,等满足了一定的条件(达到一定数量,或遇到特定字符,如换行符和文件结束符EOF),再将缓冲区中的内容一次性写入文件,这样就大大增加了文件读写的速度,但也为我们编程带来了一点点麻烦。如果有一些数据,我们认为已经写入了文件,实际上因为没有满足特定的条件,它们还只是保存在缓冲区内,这时我们用_exit()函数直接将进程关闭,缓冲区中的数据就会丢失,反之,如果想保证数据的完整性,就一定要使用exit()函数。

Exit的函数声明在stdlib.h头文件中。

_exit的函数声明在unistd.h头文件当中。

下面的实例比较了这两个函数的区别。printf函数就是使用缓冲I/O的方式,该函数在遇到\n换行符时自动的从缓冲区中将记录读出。实例就是利用这个性质进行比较的。

exit.c源码

#include<stdlib.h>#include<stdio.h>intmain(void){printf("Usingexit...\n");printf("Thisisthecontentinbuffer");exit(0);}

输出信息:

Usingexit...

Thisisthecontentinbuffer

#include<unistd.h>#include<stdio.h>intmain(void){printf("Usingexit...\n");//如果此处不加\n的话,这条信息有可能也不会显示在终端上。printf("Thisisthecontentinbuffer");_exit(0);}

则只输出:

Usingexit...

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

#include<stdio.h>;

intmain(){printf("%c",‘c‘);_exit(0);}


发表评论:

最近发表
网站分类
标签列表