使用 C 语言的 exit 函数

Jinku Hu 2023年10月12日
  1. 使用 exit 函数终止 C 语言中的进程
  2. 使用 atexit 函数在 C 语言中注册退出处理程序
使用 C 语言的 exit 函数

本文将演示如何使用 C 语言中的 exit 函数的多种方法。

使用 exit 函数终止 C 语言中的进程

当给定的程序在基于 UNIX 的操作系统上运行时,它被称为一个进程。进程可能是一个长期运行的守护进程式的程序,也可能是一个简单的命令行实用程序,但最终,它们都会在某个时间来到终止点。终止可以是由某些故障/信号引起的非正常终止,也可以是进程本身通过调用 exit 库函数作为正常行为优雅地终止。它需要一个单整数参数,指定返回给父进程的终止值。注意,exit 函数在调用进程中不返回。

exit 函数是建立在系统调用 _exit 之上的标准库函数(我们将在接下来的段落中讨论)。不过,它进行的操作还是比较多的,不仅仅是把调用者引到终止点。也就是说,exit 会对程序进行一些清理程序,比如注册为退出处理程序的函数被调用,标准 I/O 流缓冲区被刷新,然后才调用 _exit。要知道,_exit 是 UNIX 特有的系统调用,而 exit 是标准 C 库的一部分,可以在不同平台上利用。

下面的例子演示了使用 EXIT_SUCCESS 状态参数调用 exit 函数,这个参数是一个宏常量,值为 0,通常表示成功返回。

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>

int main(int argc, char *argv[]) {
  // Execute program code here

  printf("Executing the program...\n");
  sleep(5);

  //    exit(0);
  exit(EXIT_SUCCESS);
}

输出:

Executing the program...

使用 atexit 函数在 C 语言中注册退出处理程序

atexit 函数用于注册退出处理程序,这些程序只是用户实现的函数,当使用 exit 调用终止进程时,应该被调用。atexit 把类型为 void (*function)(void) 的函数指针作为唯一的参数。

请注意,多个 atexit 调用可以注册多个函数,这导致给定的函数按注册的相反顺序执行。如果调用失败,atexit 会返回一个非零值。注意,如果进程被外部信号异常终止,注册的函数可能不会被调用。下一个示例代码实现了两个 static 函数被注册为退出处理程序,在 5 秒的睡眠后,进程用 exit 调用终止。

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>

static void atexitFunc(void) { printf("atexitFunc called\n"); }

static void atexitFunc2(void) { printf("atexitFunc2 called\n"); }

int main(int argc, char *argv[]) {
  if (atexit(atexitFunc) != 0) {
    perror("atexit");
    exit(EXIT_FAILURE);
  }

  if (atexit(atexitFunc2) != 0) {
    perror("atexit");
    exit(EXIT_FAILURE);
  }

  printf("Executing the program...\n");
  sleep(5);

  exit(EXIT_SUCCESS);
}

输出:

Executing the program...
atexitFunc2 called
atexitFunc called

注意 atexitFunc2 首先被调用,然后是 atexitFunc。另外,这个程序也可以直接调用 _exit 系统调用来终止,这样可以立即终止进程。不过要注意,它不会调用用 atexit 注册的函数。相反,_exit 会关闭打开的文件描述符,这会在进程终止前造成未知的延迟。此外,我们可以使用带有所需状态值的 return 语句来引起类似 exit 函数提供的终止行为。

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>

static void atexitFunc(void) { printf("atexitFunc called\n"); }

static void atexitFunc2(void) { printf("atexitFunc2 called\n"); }

int main(int argc, char *argv[]) {
  if (atexit(atexitFunc) != 0) {
    perror("atexit");
    exit(EXIT_FAILURE);
  }

  if (atexit(atexitFunc2) != 0) {
    perror("atexit");
    exit(EXIT_FAILURE);
  }

  printf("Executing the program...\n");
  sleep(5);

  _exit(EXIT_SUCCESS);
作者: Jinku Hu
Jinku Hu avatar Jinku Hu avatar

DelftStack.com 创始人。Jinku 在机器人和汽车行业工作了8多年。他在自动测试、远程测试及从耐久性测试中创建报告时磨练了自己的编程技能。他拥有电气/电子工程背景,但他也扩展了自己的兴趣到嵌入式电子、嵌入式编程以及前端和后端编程。

LinkedIn Facebook

相关文章 - C Process