服务器之家:专注于服务器技术及软件下载分享
分类导航

PHP教程|ASP.NET教程|Java教程|ASP教程|编程技术|正则表达式|C/C++|IOS|C#|Swift|Android|VB|R语言|JavaScript|易语言|vb.net|

服务器之家 - 编程语言 - C/C++ - 浅析结束程序函数exit, _exit,atexit的区别

浅析结束程序函数exit, _exit,atexit的区别

2020-12-26 15:59C语言教程网 C/C++

在一个程序中最多可以用atexit()注册32个处理函数,这些处理函数的调用顺序与其注册的顺序相反,也即最先注册的最后调用,最后注册的最先调用

多时候我们需要在程序退出的时候做一些诸如释放资源的操作,但程序退出的方式有很多种,比如main()函数运行结束、在程序的某个地方用exit() 结束程序、用户通过Ctrl+C或Ctrl+break操作来终止程序等等,因此需要有一种与程序退出方式无关的方法来进行程序退出时的必要处理。方法就 是用atexit()函数来注册程序正常终止时要被调用的函数。

atexit()函数的参数是一个函数指针,函数指针指向一个没有参数也没有返回值的函数。atexit()的函数原型是:int atexit (void (*)(void));

在一个程序中最多可以用atexit()注册32个处理函数,这些处理函数的调用顺序与其注册的顺序相反,也即最先注册的最后调用,最后注册的最先调用。

复制代码 代码如下:


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

 

void fun()
{
    printf("fun/n");
}

int main()
{
    atexit(fun);
    printf("hello/n");
    return 0;
}

// int atexit(void (*func)()) // 见 <stdlib.h> 中定义
// {
//     func();
//     return 0;
// }


上面的代码将输出

 

hello

fun

而把红色的注释代码去掉之后,由于Interpositioning行为,重定义了库函数,使atexit仅仅表现为一个普通的函数

因此输出

复制代码 代码如下:


fun

 

hello

#include <stdlib.h>

#define EXIT_FAILURE ...
#define EXIT_SUCCESS ...

void exit (int status);
void _Exit(int status);          //C99
void abort(void);

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


exit、_Exit与abort函数使程序终止,控制并不返回到这些函数的调用者。

 

函数exit正常终止程序,并进行下列清理操作:

1.(进对标准C语言)所有想atexit函数注册的函数按与注册时相反的顺序调用,注册几次就调用几次。

2. 刷新打开的是输出流,关闭所有打开的数据流。

3. 删除tepfile函数生成的文件。

4. 控制返回宿主环境,提供状态值。

按照许多系统中的习惯,status值为0表示终止程序成功,用非0值表示异常终止。标准C语言中数值0和宏 EXIT_SCCESS的值表示终止成功,宏EXIT_FAILURE的值表示终止不成功,其他值的含义由实现定义。从函数main返回一个整数值相当于用这个值调用exit函数。

函数_Exit与exit函数不同之处在于既不调用atexit注册的退出处理器,也不调用singal注册的信号处理器。是否进行其他清理操作由实现定义,如关闭所有打开的数据流。_Exit是C99增加的,传统上有些实现用名为_exit的函数提供类似功能。

abort函数使程序异常终止,不调用向atexit注册的函数。abort是否引起清理操作由实现定义,向宿主系统返回的状态值也由实现定义,但应表示为“不成功”。在标准C语言和许多传统实现中,调用abort转换成可以捕获的特殊信号(标准C语言中为 SIGABRT)。如果信号被忽略或处理器返回,则标准C语言实现仍然终止程序,而其他实现可能使abort函数返回调用者。断言失败也会调用 abort。

atexit函数是标准C语言中增加的,它注册一个函数,使得调用exit时或函数main返回时会调用这个函数。程序异常终止时(如用abort或raise终止),不调用注册的函数。实现应允许至少注册32个函数。如果注册成功,则atexit函数返回0,否则返回非0值,函数无法注销。所有向atexit函数注册的函数按与注册相反的顺序调用,然后再由atexit函数完成所有标准清理操作。每个函数不代参数调用,应具有返回类型void。注册函数不能引用任何不是自己定义的存储类为auto或 register的对象(例如通过指针引用)。函数注册几次就会在此时调用几次。

指针函数使用示例:

复制代码 代码如下:


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

 

typedef void (*pFunc)(float a);

// int atexit(void (*func)())
// {
//     func();
//     return 0;
// }

int atexitf( void (*func)(float),float a)
{
    func(a);
    return 0;
}

void test(float a)
{
     printf("test %f/n",a);
}

// void fun()
// {
//     printf("fun/n");
// }

int main()
{
    pFunc pFunc1 = (pFunc)test;  // 函数指针赋值
    pFunc1(4.5);
/*    atexit(fun);*/
    atexitf(pFunc1,3.66);
    atexitf(test,3.66);
    //atexitf(test(4.3));
    printf("hello/n");
    getchar();
    return 0;
}

 

延伸 · 阅读

精彩推荐