exit、return、_exit、_Exit这几个函数的区别

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

1exit用于在程序运行的过程中随时结束程序,其参数是返回给OS。也可以这么讲:exit函数是退出应用程序,并将应用程序的一个状态返回给OS,这个状态标识了应用程序的一些运行信息。

main函数结束时也会隐式地调用exit函数,exit函数运行时首先会执行由atexit()函数登记的函数,然后会做一些自身的清理工作,同时刷新所有输出流、关闭所有打开的流并且关闭通过标准I/O函数tmpfile()创建的临时文件。

exit是系统调用级别的,它表示了一个进程的结束,它将删除进程使用的内存空间,同时把错误信息返回父进程。通常情况:exit(0)表示程序正常, exit(1)exit(-1)表示程序异常退出,exit(2)表示系统找不到指定的文件。

2return是语言级别的,它表示了调用堆栈的返回;return是返回函数值并退出函数,通常0为正常退出,0为非正常退出,请注意,如果是在主函数main,自然也就结束当前进程了(也就是说,在main()里面,你可以用return n,也能够直接用exit(n)来做),如果不是在main函数中,那就是退回上一层调用。在多个进程时,如果有时要检测上个进程是否正常退出,就要用到上个进程的返回值。

2、进程环境与进程控制

如果exit(int n)是在main函数里被调用的,那么exit(int n)就直接退出程序,并返回一个int型的值。一般在shell下面,运行一个程序,然后使用命令echo $?就能得到该程序的返回值,也就是退出值。理论上exit可以返回小于256的任何整数,返回的不同数值主要是给调用者作不同处理的。

对于单独的进程exit的返回值是返回给操作系统的,但如果是多进程,则是返回给父进程的。父进程里面调用waitpid()等函数得到子进程退出的状态,以便作不同处理。根据相应的返回值来让调用者作出相应的处理。

总的说来,exitint n)就是当前进程把其控制权返回给调用该子程序的主程序,括号里的是返回值,告诉调用程序该程序的运行状态。

1)进程的开始:

C程序是从main函数开始执行,原型如下: int main(int argc, char *argv[]);通常main的返回值是int,正确返回0

2)进程终止:

C程序的终止分为两种:正常终止和异常终止

正常终止分为: return, exit, _exit, _Exit, pthreade_exit

异常中指分为: abort, SIGNAL,线程响应取消

主要说一下正常终止的前4,exit系列函数.

#include <stdlib.h>  

void exit(int status);

void _Exit(int status);

#include <unistd.h>  

void _exit(int status);

以上3个函数的区别是:

exit()(return 0)调用终止处理程序用户空间的标准I/O清理程序(fclose),_exit_Exit不调用而直接由内核接管进行清理因此,main函数中exit(0)等价于return 0.

3atexit终止处理程序:

ISO C规定,一个进程最多可登记32个终止处理函数,这些函数由exit登记相反的顺序自动调用。如果同一函数登记多次,也会被调用多次。

原型如下:

#include <stdlib.h>

int atexit(void (*func)(void));

其中参数是一个函数指针,指向终止处理函数,该函数无参无返回值atexit函数本身成功调用后返回0

以下面的程序为例:

#include <stdlib.h>

static void myexit1()

{

    printf("first exit handler\n");

}

static void myexit2()

{

    printf("second exit handler\n");

}

int main()

{

    atexit(my_exit2)

  atexit(my_exit1)

atexit(my_exit1)

    printf("main is done\n");

    return 0;// 相当于exit(0)

}

运行结果:

$ ./a.out

main is done

first exit handler

first exit handler

second exit handler

main函数结束时也会隐式地调用exit函数,exit函数运行时首先会执行由atexit()函数登记的函数,然后会做一些自身的清理工作,同时刷新所有输出流关闭所有打开的流并且关闭通过标准I/O函数tmpfile()创建的临时文件

注意上面程序的结果,可以发现这些函数由exit按登记相反的顺序自动调用(先myexit1myexit2)。如果同一函数登记多次,也会被调用多次(如这里的myexit1)。

而这些处理函数都是在程序退出的时候利用atexit函数调用了这些处理函数。但是如果用_exit()退出程序,则它不关闭任何文件,不清除任何缓冲器、也不调用任何终止函数!

 

#include <stdlib.h>

exit和_exit函数用于正常终止一个程序: _exit立即进入内核,exit则先执行一些清除处理(包括调用执行各终止处理程序,关闭所有标准I / O流等),然后进入内核。使用不同头文件的原因是:exit是由ANSI C说明的,而_exit则是由POSIX.1说明的。
由于历史原因,exit函数总是执行一个标准I/O库的清除关闭操作:对于所有打开流调用 fclose 函数。exit和_exit都带一个整型参数,称之为终止状态(exit status)。大多数UNIX shell都提供检查一个进程终止状态的方法。如果( a )若调用这些函数时不带终止状态,或( b ) main执行了一个无返回值的re turn语句,或( c ) main执行隐式返回,则该进程的终止状态是末定义的。这就意味着,下列经典性的C语言程序:
#include <stdio.h>
main ()
{
printf ("hello, world \n");
}
是不完整的,因为main函数没有使用return语句返回(隐式返回),它在返回到C的起动例程时并没有返回一个值(终止状态)。另外,若使用:
return( 0 ) ;或者
exit( 0 );
则向执行此程序的进程(常常是一个shell进程)返回终止状态0。另外,main函数的说明实际上应当是:
int main(void)
将main说明为返回一个整型以及用exit代替return,对某些C编译程序和UNIX lint(1)程序而言会产生不必要的警告信息,因为这些编译程序并不了解main中的exit与return语句的作用相同。警告信息可能是“ control reaches end of nonvoid function(控制到达非void函数的结束处)”。避开这种警告信息的一种方法是:在main中使用return语句而不是exit。但是这样做的结果是不能用UNIX的grep公用程序来找出程序中所有的exit调用。另外一个解决方法是将main说明为返回void而不是int,然后仍旧调用exit。这也避开了编译程序的警告,但从程序设计角度看却并不正确。本章将main表示为返回一个整型,因为这是ANSIC和POSIX.1所定义的。我们将不理会编译程序不必要的警告。
atexit函数
按照ANSI C的规定,一个进程可以登记多至32个函数,这些函数将由exit自动调用。我们称这些函数为终止处理程序(exit handler),并用atexit函数来登记这些函数。
#include <stdlib.h>
int atexit(void *( func) ( void ) ) ;
返回:若成功则为0,若出错则为非0
其中, atexit的参数是一个函数地址,当调用此函数时无需向它传送任何参数,也不期望它返回一个值。exit以登记这些函数的相反顺序调用它们。同一函数如若登记多次,则也被调用多次。
终止处理程序这一机制由ANSI C最新引进。S V R 4和4 . 3 + B S D都提供这种机制。系统V的早期版本和4 . 3 B S D则都不提供此机制。
根据ANSI C和POSIX.1,exit首先调用各终止处理程序,然后按需多次调用fclose,关闭所有打开流。图中显示了一个C程序是如何起动的,以及它终止的各种方式。注意,内核使程序执行的唯一方法是调用一个e x e c函数。进程自愿终止的唯一方法是显式或隐式地(调用e x i t )调用_ e x i t。进程也可非自愿地由一个信号使其终止。

进程有三种正常终止法及两种异常终止法。
(1) 正常终止:
(a) 在m a i n函数内执行r e t u r n语句。如在7 . 3节中所述,这等效于调用e x i t。
(b) 调用e x i t函数。此函数由ANSI C定义,其操作包括调用各终止处理程序(终止处理程序在调用a t e x i t函数时登录),然后关闭所有标准I / O流等。因为ANSI C并不处理文件描述符、多进程(父、子进程)以及作业控制,所以这一定义对U N I X系统而言是不完整的。
(c) 调用_ e x i t系统调用函数。此函数由e x i t调用,它处理U N I X特定的细节。_ e x i t是由P O S I X . 1说明的。
(2) 异常终止:
(a) 调用a b o r t。它产生S I G A B RT信号,所以是下一种异常终止的一种特例。
(b) 当进程接收到某个信号时。(第1 0章将较详细地说明信号。)进程本身(例如调用a b o r t函数)、其他进程和内核都能产生传送到某一进程的信号。例如,进程越出其地址空间访问存储单元,或者除以0,内核就会为该进程产生相应的信号。不管进程如何终止,最后都会执行内核中的同一段代码。这段代码为相应进程关闭所有打
开描述符,释放它所使用的存储器等等。对上述任意一种终止情形,我们都希望终止进程能够通知其父进程它是如何终止的。对于e x i t和_ e x i t,这是依靠传递给它们的退出状态( exit status)参数来实现的。在异常终止情况,内核(不是进程本身)产生一个指示其异常终止原因的终止状态( termination status)。在任意一种情况下,该终止进程的父进程都能用w a i t或w a i t p i d函数(在下一节说明)取得其终止状态。
注意,这里使用了“退出状态”(它是传向e x i t或_ e x i t的参数,或m a i n的返回值)和“终止
状态”两个术语,以表示有所区别。在最后调用_ e x i t时内核将其退出状态转换成终止状态。

/************************************************************

摘要:本文详细讲述了几个出错处理的函数abort、exit、atexit、strerror函数的使用方法,并给出来具体的示例程序。


函数名: abort
功 能: 异常终止一个进程
用 法: void abort(void);
头文件:#include <stdlib.h>
说明:abort函数是一个比较严重的函数,当调用它时,会导致程序异常终止,
而不会进行一些常规的清除工作,比如释放内存等。
程序例:
#include <stdio.h>
#include <stdlib.h>

int main(void)
{
puts( "About to abort....\n" );
abort();

puts( "This will never be executed!\n" );
exit( EXIT_SUCCESS );
}

[root@localhost error_process]# gcc abort.c
[root@localhost error_process]# ./a.out
About to abort....

已放弃


表头文件 #include<stdlib.h>
定义函数 void exit(int status);

exit()用来正常终结目前进程的执行,并把参数 status 返回给父进程,
而进程所有的缓冲区数据会自动写回并关闭未关闭的文件。
它并不像abort那样不做任何清理工作就退出,而是在完成所有的清理工作后才退出程序。


atexit(设置程序正常结束前调用的函数)

表头文件 #include<stdlib.h>
定义函数 int atexit (void (*function)(void));

atexit()用来设置一个程序正常结束前调用的函数。当程序通过调
用 exit()或从 main 中返回时,参数 function 所指定的函数会先被
调用,然后才真正由 exit()结束程序。

返回值 如果执行成功则返回 0,否则返回-1,失败原因存于 errno 中。

#include <stdlib.h>
#include <stdio.h>

void my_exit(void)
{
printf( "Before exit....\n" );
}

int main(void)
{
atexit( my_exit );
return 0;
}

[root@localhost error_process]# gcc atexit.c
[root@localhost error_process]# ./a.out
Before exit....



strerror(返回错误原因的描述字符串)

表头文件 #include<string.h>
定义函数 char * strerror(int errnum);
strerror() 用来依参数 errnum 的错误代码来查询其错误原因的描述字符串,然后将该字符串指针返回。
这时如果把 errno 传个strerror,就可以得到可读的提示信息,而不再是一个冷冰冰的数字了。

返回值 返回描述错误原因的字符串指针。

#include <string.h>
#include <stdio.h>

int main(void)
{
int i;

for ( i=0; i<10; i++ )
{
  printf( "%d:%s\n", i, strerror(i) );
}

return 0;
}

[root@localhost error_process]# gcc strerror.c
[root@localhost error_process]# ./a.out
0:Success
1:Operation not permitted
2:No such file or directory
3:No such process
4:Interrupted system call
5:Input/output error
6:No such device or address
7:Argument list too long
8:Exec format error
9:Bad file descriptor
[root@localhost error_process]#
  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
`__enter__`和`__exit__`是Python中的特殊方法,用于实现上下文管理器。 一个上下文管理器可以被用于处理资源的获取和释放,例如文件、锁或数据库连接。当使用`with`语句时,会调用上下文管理器的`__enter__`方法获取资源并返回一个相关的对象,然后执行`with`语句块中的代码。最后,无论代码块是否抛出异常,都会调用上下文管理器的`__exit__`方法释放资源。 `__enter__`方法通常返回一个对象,这个对象会被赋值给`as`关键字指定的变量。例如: ```python with open("example.txt") as file: # do something with file ``` 在这个例子中,`open`函数返回一个文件对象,它实现了上下文管理器接口。`__enter__`方法会被调用以打开文件并返回文件对象,然后文件对象会被赋值给`file`变量。在`with`语句块中,我们可以通过`file`变量访问文件对象。最后,`__exit__`方法会被调用以关闭文件。 `__exit__`方法接受三个参数:异常类型、异常对象和异常回溯。如果代码块正常退出,这些参数都为`None`。如果代码块抛出了异常,这些参数包含有关异常的信息。`__exit__`方法应该处理异常并释放资源,然后返回`True`表示异常已被处理。如果`__exit__`方法返回`False`,则异常会继续传播。 例如,下面是一个简单的上下文管理器,它可以用于计时: ```python import time class Timer: def __enter__(self): self.start = time.time() return self def __exit__(self, exc_type, exc_value, traceback): elapsed_time = time.time() - self.start print(f"Elapsed time: {elapsed_time:.2f} seconds") return True ``` 使用这个上下文管理器,我们可以在代码块运行时计时: ```python with Timer(): time.sleep(2) ``` 这个代码块会睡眠2秒,并在完成后输出经过的时间。在这个例子中,`__enter__`方法会记录当前时间,并返回`self`对象。在`with`语句块中,我们可以访问`self`对象并执行任意代码。最后,`__exit__`方法会计算经过的时间并输出它。如果代码块抛出异常,`__exit__`方法会被调用以释放资源。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值