当前位置:  编程技术>c/c++/嵌入式

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

    来源: 互联网  发布时间:2014-10-19

    本文导语:  多时候我们需要在程序退出的时候做一些诸如释放资源的操作,但程序退出的方式有很多种,比如main()函数运行结束、在程序的某个地方用exit() 结束程序、用户通过Ctrl+C或Ctrl+break操作来终止程序等等,因此需要有一种与程序...

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

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

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

代码如下:

#include
#include

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

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

// int atexit(void (*func)()) // 见 中定义
// {
//     func();
//     return 0;
// }


上面的代码将输出

hello

fun

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

因此输出

代码如下:

fun

hello

#include

#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
#include

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;
}


    
 
 
 
本站(WWW.)旨在分享和传播互联网科技相关的资讯和技术,将尽最大努力为读者提供更好的信息聚合和浏览方式。
本站(WWW.)站内文章除注明原创外,均为转载、整理或搜集自网络。欢迎任何形式的转载,转载请注明出处。












  • 相关文章推荐
  • jquery重复提交请求的原因浅析
  • 浅析删除表的几种方法(delete、drop、truncate)
  • 浅析Oracle中char和varchar2的区别
  • 浅析java中print和println的区别
  • 浅析java class 文件
  • 求教,linux内核代码关于网络部分的浅析
  • 浅析如何在tnsnames.ora中配置监听
  • 浅析ThinkPHP中execute和query方法的区别
  • 浅析C#的复制和克隆
  • 浅析jQuery中调用ajax方法时在不同浏览器中遇到的问题
  • 浅析jquery的作用与优势
  • 浅析SQL server 临时表
  • 浅析c++ 宏 #val 在unicode下的使用
  • 浅析c#中如何在form的webbrowser控件中获得鼠标坐标
  • 浅析SQL语句中GROUP BY的用法
  • PHP强制下载文件方法浅析
  • 浅析打开eclipse出现Incompatible JVM的解决方法
  • 浅析SQL存储过程和事务处理
  • C#访问权限修饰符浅析
  • 浅析Java方法传值和传引用问题


  • 站内导航:


    特别声明:169IT网站部分信息来自互联网,如果侵犯您的权利,请及时告知,本站将立即删除!

    ©2012-2021,,E-mail:www_#163.com(请将#改为@)

    浙ICP备11055608号-3