使用 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);
Founder of DelftStack.com. Jinku has worked in the robotics and automotive industries for over 8 years. He sharpened his coding skills when he needed to do the automatic testing, data collection from remote servers and report creation from the endurance test. He is from an electrical/electronics engineering background but has expanded his interest to embedded electronics, embedded programming and front-/back-end programming.
LinkedIn