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

所属分类: 软件编程 / C 语言 阅读数: 96
收藏 0 赞 0 分享

多时候我们需要在程序退出的时候做一些诸如释放资源的操作,但程序退出的方式有很多种,比如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;
}

更多精彩内容其他人还在看

C++中四种对象生存期和作用域以及static的用法总结分析

以下是对C++中四种对象生存期和作用域以及static的用法进行了详细的介绍,需要的朋友可以过来参考下
收藏 0 赞 0 分享

C++嵌套类与局部类详细解析

从作用域的角度看,嵌套类被隐藏在外围类之中,该类名只能在外围类中使用。如果在外围类之外的作用域使用该类名时,需要加名字限定
收藏 0 赞 0 分享

C++空类详解

以下是对C++中的空类进行了详细的介绍,需要的朋友可以过来参考下
收藏 0 赞 0 分享

C++之友元:友元函数和友元类详解

友元是一种允许非类成员函数访问类的非公有成员的一种机制。可以把一个函数指定为类的友元,也可以把整个类指定为另一个类的友元
收藏 0 赞 0 分享

C++中返回指向函数的指针示例

int (*ff(int)) (int *,int);表示:ff(int)是一个函数,带有一个int型的形参,该函数返回int (*) (int *,int),它是一个指向函数的指针,所指向的函数返回int型并带有两个分别是Int*和int型的形参
收藏 0 赞 0 分享

C数据结构之单链表详细示例分析

以下是对C语言中的单链表进行了详细的分析介绍,需要的朋友可以过来参考下
收藏 0 赞 0 分享

C数据结构之双链表详细示例分析

以下是对c语言中的双链表进行了详细的分析介绍,需要的朋友可以过来参考下
收藏 0 赞 0 分享

浅析如何在c语言中调用Linux脚本

如何在c语言中调用Linux脚本呢?下面小编就为大家详细的介绍一下吧!需要的朋友可以过来参考下
收藏 0 赞 0 分享

深入解析unsigned int 和 int

以下是对unsigned int和int进行了详细的分析介绍,需要的朋友可以过来参考下
收藏 0 赞 0 分享

浅谈C++中的string 类型占几个字节

本篇文章小编并不是为大家讲解string类型的用法,而是讲解我个人比较好奇的问题,就是string 类型占几个字节
收藏 0 赞 0 分享
查看更多