exit和_exit的区别

本文详细探讨了在C/C++程序中使用exit()与_exit()函数的区别,特别是在涉及fork()和vfork()的情况下。exit()函数会在进程退出前执行清理操作和调用用户自定义的清理程序,而_exit()仅执行内核级别的清理。在特定场景下,不当使用exit()可能导致意外的行为。
exit()在结束调用它的进程之前,要进行如下步骤:
1.cleanup();
2.在atexit()注册的函数;
最后调用_exit()函数。。。


‘exit()’与‘_exit()’有不少区别在使用‘fork()’,特别是‘vfork()’时变得很
突出。

‘exit()’与‘_exit()’的基本区别在于前一个调用实施与调用库里用户状态结构
(user-mode constructs)有关的清除工作(clean-up),而且调用用户自定义的清除程序
(译者注:自定义清除程序由atexit函数定义,可定义多次,并以倒序执行),相对
应,后一个函数只为进程实施内核清除工作。

在由‘fork()’创建的子进程分支里,正常情况下使用‘exit()’是不正确的,这是
因为使用它会导致标准输入输出(译者注:stdio: Standard Input Output)的缓冲区被
清空两次,而且临时文件被出乎意料的删除(译者注:临时文件由tmpfile函数创建
在系统临时目录下,文件名由系统随机生成)。在C++程序中情况会更糟,因为静
态目标(static objects)的析构函数(destructors)可以被错误地执行。(还有一些特殊情
况,比如守护程序,它们的*父进程*需要调用‘_exit()’而不是子进程;适用于绝
大多数情况的基本规则是,‘exit()’在每一次进入‘main’函数后只调用一次。)

在由‘vfork()’创建的子进程分支里,‘exit()’的使用将更加危险,因为它将影响
*父*进程的状态。
在 C 语言中,`exit()` `_exit()` 都是用于**终止进程**的函数,但它们在行为使用场景上有一些关键区别。理解这些区别对于编写健壮的系统级程序非常重要。 --- ## ✅ `exit()` 函数 ### 函数原型: ```c #include <stdlib.h> void exit(int status); ``` ### 功能: - **正常终止当前进程**。 - 在退出前,会执行一些**清理操作**: - 调用通过 `atexit()` 或 `on_exit()` 注册的清理函数; - 刷新所有打开的 `stdio` 缓冲区(如 `printf` 的缓冲区); - 关闭所有标准 I/O 流; - 将控制权交还给操作系统。 ### 示例: ```c printf("Before exit"); exit(0); // 输出会被刷新 ``` --- ## ✅ `_exit()` 函数 ### 函数原型: ```c #include <unistd.h> void _exit(int status); ``` ### 功能: - **立即终止当前进程**,不执行任何清理操作。 - 不调用 `atexit()` 注册的函数; - 不刷新 `stdio` 缓冲区; - 不关闭文件流。 ### 示例: ```c printf("Before _exit"); _exit(0); // "Before _exit" 可能不会输出 ``` --- ## ✅ `exit()` 与 `_exit()` 的主要区别 | 特性 | `exit()` | `_exit()` | |------|----------|-----------| | 所属头文件 | `<stdlib.h>` | `<unistd.h>` | | 是否刷新缓冲区 | 是 | 否 | | 是否调用清理函数 | 是 | 否 | | 是否关闭文件流 | 是 | 否 | | 更适合用于 | 正常退出 | 子进程退出(如 fork 后的子进程) | --- ## ✅ 在 `fork()` 后使用 `exit()` `_exit()` 的注意事项 在 `fork()` 创建的子进程中,通常建议使用 `_exit()` 而不是 `exit()`,原因如下: - 子进程继承了父进程的 `stdio` 缓冲区内容; - 如果使用 `exit()`,会刷新缓冲区,可能导致数据被重复输出; - `_exit()` 直接终止进程,避免这种副作用。 ### 示例: ```c pid_t pid = fork(); if (pid == 0) { // 子进程 printf("子进程输出"); _exit(0); // 不刷新缓冲区,可能导致输出不完整 } ``` 如果希望子进程也刷新缓冲区并执行清理函数,可以使用 `exit()`。 --- ## ✅ 返回状态码 `status` - `exit(0)` 表示程序正常退出; - `exit(EXIT_SUCCESS)` 与 `exit(0)` 等价; - `exit(EXIT_FAILURE)` 表示异常退出; - 状态码会传递给父进程(通过 `wait()` 或 `waitpid()` 获取)。 --- ## ✅ 总结一句话: > `exit()` 是“优雅退出”,适合主进程;`_exit()` 是“立即退出”,适合子进程,避免缓冲区重复刷新。 --- ##
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值