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

所属分类: 软件编程 / C 语言 阅读数: 112
收藏 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++实现string与各种类型之间的转换

这个类在头文件中定义, < sstream>库定义了三种类:istringstream、ostringstream和stringstream,分别用来进行流的输入、输出和输入输出操作。另外,每个类都有一个对应的宽字符集版本
收藏 0 赞 0 分享

C++如何通过ostringstream实现任意类型转string

再使用整型转string的时候感觉有点棘手,因为itoa不是标准C里面的,而且即便是有itoa,其他类型转string不是很方便。后来去网上找了一下,发现有一个好方法
收藏 0 赞 0 分享

C/C++指针小结

要搞清一个指针需要搞清指针的四方面的内容:指针的类型,指针所指向的类型,指针的值或者叫指针所指向的内存区,还有指针本身所占据的内存区
收藏 0 赞 0 分享

C++ 类的静态成员深入解析

在C++中类的静态成员变量和静态成员函数是个容易出错的地方,本文先通过几个例子来总结静态成员变量和成员函数使用规则,再给出一个实例来加深印象
收藏 0 赞 0 分享

C++类的静态成员初始化详细讲解

通常静态数据成员在类声明中声明,在包含类方法的文件中初始化.初始化时使用作用域操作符来指出静态成员所属的类.但如果静态成员是整型或是枚举型const,则可以在类声明中初始化
收藏 0 赞 0 分享

C++类静态成员与类静态成员函数详解

静态成员不可在类体内进行赋值,因为它是被所有该类的对象所共享的。你在一个对象里给它赋值,其他对象里的该成员也会发生变化。为了避免混乱,所以不可在类体内进行赋值
收藏 0 赞 0 分享

C++中的friend友元函数详细解析

友元可以是一个函数,该函数被称为友元函数;友元也可以是一个类,该类被称为友元类。友元函数的特点是能够访问类中的私有成员的非成员函数。友元函数从语法上看,它与普通函数一样,即在定义上和调用上与普通函数一样
收藏 0 赞 0 分享

static全局变量与普通的全局变量的区别详细解析

以下是对static全局变量与普通的全局变量的区别进行了详细的分析介绍,需要的朋友可以过来参考下,希望对大家有所帮助
收藏 0 赞 0 分享

C++ explicit关键字的应用方法详细讲解

C++ explicit关键字用来修饰类的构造函数,表明该构造函数是显式的,既然有"显式"那么必然就有"隐式",那么什么是显示而什么又是隐式的呢?下面就让我们一起来看看这方面的知识吧
收藏 0 赞 0 分享

教你5分钟轻松搞定内存字节对齐

随便google一下,人家就可以跟你解释的,一大堆的道理,我们没怎么多时间,讨论为何要对齐.直入主题,怎么判断内存对齐规则,sizeof的结果怎么来的,请牢记以下3条原则
收藏 0 赞 0 分享
查看更多