美文网首页Linux学习之路我用 Linux剑指BAT
C语言深度总结[全面认识main函数之后运行代码]

C语言深度总结[全面认识main函数之后运行代码]

作者: batbattle | 来源:发表于2018-06-15 07:44 被阅读9次

    一、main结束不代表整个进程结束

    (1)全局对象的析构函数会在main函数之后执行;

     (2)用atexit注册的函数也会在main之后执行。

    二、用atexit注册的函数会在main结束之后执行

    #include<stdio.h>

    #include<stdlib.h>

    void fn1(void)

    {

    printf("next.\n");

    }

    void fn2(void)

    {

    printf("executed ");

    }

    void fn3(void)

    {

    printf("is ");

    }

    void fn4(void)

    {

    printf("This ");

    }

    int main(void)

    {

    //

    // 注册需要在 main 函数结束后执行的函数. 

    // 请注意它们的注册顺序和执行顺序

    // 在 main 函数结束后被调用,调用顺序与注册顺序相反。 先注册后执行。

    //

    atexit(fn1);

    atexit(fn2);

    atexit(fn3);

    atexit(fn4);

    // 这条输出语句具有参照性,它可不是最后一句输出.

    puts("This is executed first.");

    // EXIT_SUCCESS 代表 0,它定义在 stdlib.h 中。第一节课里也介绍了

    puts("老铁,感谢有缘一起学习C语言,除了跟着这套课程走,为了更好地解决大家实操中的问题,可以加QQ群676593534,以便及时交流。群里还有很多精致资料哦,我这这里等你!");

    return EXIT_SUCCESS;

    }

    关于atexit函数稍微补充一下,原型如下:

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

      atexit 函数是标准 C 新增的。它“注册”一个函数,使这个函数将在 exit 函数被调用时或者当 mian 函数返回时被调用。当程序异常终止时(例如调用 abort 或 raise),通过它注册的函数并不会被调用。编译器必须至少允许程序员注册32个函数。如果注册成功,atexit 返回0,否则返回非零值。

    没有办法取消一个函数的注册。在 exit 所执行的任何标准清理操作之前,被注册的函数按照与注册顺序相反的顺序被依次调用。每个被调用的函数不接受任何参数,并且返回类型是 void。

    被注册的函数不应该试图引用任何存储类别为 auto 或 register 的对象(例如通过指针),除非是它自己所定义的。多次注册同一个函数将导致这个函数被多次调用。

       atexit是注册后进先出的函数,和函数入栈出栈是一样的。

       在这里注册了四个函数,理解为入栈的顺序为fn1() -> fn2() -> fn3() -> fn4();出栈的顺序正好相反,而什么时候出栈呢?就是在调用函数结束时,准确的说应该是函数调用的最后的操作就是出栈过程。main()同样也是一个函数,在结束时,按出栈的顺序调用四个函数,即为fn4() -> fn3() -> fn2() -> fn1();

    Demo演示

    如来在手,bug远去

    P.S简书这个不友好的编辑器,无法把我心中的如来佛祖完美演绎出来。想把佛祖请到你家,给我留言即可

    #include<stdio.h>

    #include<stdlib.h>

    /**

     *                    _ooOoo_

     *                  o8888888o

     *                  88" . "88

     *                  (| -_- |)

     *                    O\ = /O

     *                ____/`---'\____

     *              .  ' \\| |// `.

     *              / \\||| : |||// \

     *            / _||||| -:- |||||- \

     *              | | \\\ - /// | |

     *            | \_| ''\---/'' | |

     *              \ .-\__ `-` ___/-. /

     *          ___`. .' /--.--\ `. . __

     *        ."" '< `.___\_<|>_/___.' >'"".

     *      | | : `- \`.;`\ _ /`;.`/ - ` : | |

     *        \ \ `-. \_ __\ /__ _/ .-` / /

     * ======`-.____`-.___\_____/___.-`____.-'======

     *                    `=---='

     *

     * .............................................

     *          佛祖保佑            永无BUG

     */

    void fn0( void ), fn1( void ), fn2( void ), fn3( void ), fn4( void );

    int main( void )

    {

    atexit( fn0 );

    atexit( fn1 );

    atexit( fn2 );

    atexit( fn3 );

    atexit( fn4 );

    printf( "This is executed first.\n" );

        printf("main will quit now!\n");

    return 0;

    }

    void fn0()

    {

    printf( "哎呦,我的小心脏啊,吓死宝宝了。竟然还真可以这样玩,扶我起来,我要继续学习!\n" );

    }

    void fn1()

    {

    printf( "next.\n" );

    }

    void fn2()

    {

    printf( "executed " );

    }

    void fn3()

    {

    printf( "is " );

    }

    void fn4()

    {

    printf( "This " );

    }

    总结:

    1.main函数不是可运行的最后一个函数。

    2.由于代码在任何一个地方都有可能退出(主动和被动),通过atexit可以注册回调清理函数。注册这个函数的目的就是为了在函数退出时调用的,即使是main()函数也是这样的。可以在这些函数中加入一些清理工作,比如内存释放、关闭打开的文件、关闭socket描述符、释放锁等等。

    3.atexit是注册后进先出的函数,和函数入栈出栈是一样的。

    注:这里的主动退出是指,程序调用诸如exit、_exit、pthread_exit等函数。被动退出是指,发生不可预知的异常导致,如访问非法内存,访问越界,OOM(out of memery),除零,类型错误等等。

    相关文章

      网友评论

        本文标题:C语言深度总结[全面认识main函数之后运行代码]

        本文链接:https://www.haomeiwen.com/subject/rkhpeftx.html