_exit系统调用

进程的终止

  • _exit()

    通常进程有两种终止方式,一种是异常终止,当进程接受到某一个信号,而该信号的默认动作又是终止当前进程(term | core)时,进程就会被异常终止

    此外,进程可以使用_exit()系统调用来正常终止

    #include<unistd.h>
    void _exit(int status);   
                                      //return 0 表示正常退出,非0表示异常退出
    

    参数status定义了进程的终止状态,父进程可以使用wait来获取该状态

    status虽然为int类型,但是只有低8位会被父进程使用,还可以返回常量:EXIT_SUCCESS,EXIT_FAILURE来表示返回的状态

    虽然status的取值范围为0~255(2^8 - 1),但是最好不要使用128以上的返回值,因为当使用信号来终止进程时,shell会将变量$?(上一个命令的返回结果,0表示没有错误)置为128 + 信号值,可能会与返回值相等,造成冲突

  • exit()

    exit是标准C语言函数库,其对_exit()进行了封装

    #include<stdlib.h>
    void exit(int status)
                                      //return 0表示正常退出,非0表示异常退出
    

    使用exit时会执行的动作如下:

    • 调用退出处理程序(看下文)
    • 刷新stdio缓冲区
    • 执行_exit(status)

    顺便介绍一下return 关键字,当程序使用return关键字或者在void函数中执行到程序末尾时,会隐式地执行exit(),

    return n ------->exit(n), 没有指定返回值的话的话就是exit(0)(也不一定,C99这样规定,但是C89未做定义)

  • 退出处理程序

    退出处理程序是一个由程序设计者提供的函数,可与进程声明周期的任意时点注册,并且会在进程调用exit()时自动调用

    如果程序直接调用_exit()或者是因为信号而导致异常终止,那么信号处理程序便不会执行

    可以采用建立信号处理程序,在信号处理程序中设置标志位,进而在主程序中使用退出处理程序,来弥补一些限制

    注册退出处理程序

    • atexit()
    #include<stdlib.h>
    int atexit(void (*func)(void))
                                              //return 0 成功,非0失败
    

    atexit会将func添加到一个函数列表,进程终止时会调用该函数列表中的所有函数、

    1. 当注册多个退出处理程序时,这些函数的执行顺醋与注册顺序相反

    2. 在调用某一个退出处理程序时,如果调用发生异常导致该函数无法返回,那么剩余的所有处 理函数均不会执行

    3. 通过fork调用创建的子进程会继承附近产注册的退出处理函数(复制文本段)

      通过exec调用会移除所有已经注册的退出处理程序(初始化所有段)

    • on_exit()

      这是一个glibc非标准函数,但是功能更加强大

      #define _DEFAULT_SOURCE
      #include<stdlib.h>
      int on_exit(void(*func)(int, void*), void* arg)
                                                  //return 0 成功,非0失败
      

      func函数中的int参数为调用exit(status)时给定的status参数,void*类型参数为注册时(即使用on_exit(func, arg)),用户自己给定的参数arg,用于在退出处理程序实现一些自定义功能

  • fork(), stdio缓冲区,以及_exit()之间的交互

    由于stdio缓冲区是在进程的用户空间进行维护的,所以使用fork时就会理所当然的将其拷贝到子进程的用户空间,在父子进程调用exit()会刷新各自的stdio缓冲区,由此可能就会导致重复的输出结果

    如何避免?

    • 可以在调用fork()之前使用函数fflush()来手动刷新缓冲区,当然也可以使用setvbuf或者setbuf来关闭缓冲功能
    • 子进程使用_exit()终止而不是exit(),这样就不会刷新exit()缓冲区
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容