exit函数和_exit函数的区别

本文详细介绍了进程终止中的exit函数与_exit函数的区别,包括它们的使用场景、功能特点及代码实现方式。并通过实例展示了这两种函数在清理缓冲区方面的不同表现。

在谈论exit函数与_exit函数之前,我们先了解一下他们的使用场景,他们是在进程终止中使用的,那什么是进程终止呢?

进程终止

进程终止的场景:

(1)代码运行完毕,结果正确;
即就是进程中的所有代码都已执行完毕,而且运行结果也是你想要得到的,就像司马懿的一生,顺顺利利的结束,实现了自己的人生理想,最后也成为了千古名人
(2)代码运行完毕,结果不正确;
即就是进程中的所有代码都被执行,但是运行结果缺不是你想要得到的,就像诸葛亮一样,他的一生也算是很顺利的结束了,但是他却没有实现自己的人生抱负,给自己的人生留下了很大的遗憾。
(3)代码异常终止
即就是,代码没有被执行完,程序就结束了。就像小霸王孙策,英年早逝,没有到年龄就已过世,更不用说他有没有实现自己的梦想。

exit函数

头文件:#include<stdlib.h>和#include<unistd.h>

函数原型:void exit(int status)
//status定义了进程的终止状态,父进程通过wait来获取该值
ps:虽然status是int,但是仅有低8位可以被父进程所用,
   所以_exit(-1_时,在终端执行$?发现返回值是255

功能:直接使进程停止运行,清除其使用的内存空间,并清
     除其在内核的各种数据结构。而且在执行退出之前执行
     用户定义的清理函数,冲刷缓冲,关闭流等

这里写图片描述

简单实现exit函数

代码:

#include<stdio.h>
#include<unistd.h>
#include<stdlib.h>
int main()
{
  printf("hello world.");
  exit(0);
  printf("hahaha");
}

运行结果:

-bash-4.1$ ./a.out 
hello world.-bash-4.1$

ps: exit()函数会在终止进程前,将缓冲I/O内容清理掉,所以即使printf()里面没有”\n”也会被打印出来

_exit函数

头文件:#include<unistd.h>

函数原型:void _exit(int status)
//status定义了进程的终止状态,父进程通过wait来获取该值
ps:虽然status是int,但是仅有低8位可以被父进程所用,
   所以_exit(-1_时,在终端执行$?发现返回值是255

功能:直接使进程停止运行,清除其使用的内存
     空间,并清除其在内核的各种数据结构。

简单实现_exit函数

代码:

#include<stdio.h>
#include<unistd.h>
int main()
{
   printf("hello world.");
   _exit(0);
   printf("hahaha");
}

运行结果:

-bash-4.1$ ./a.out 
-bash-4.1$

ps: 由运行结果可看出,_exit()函数是直接终止进程,并未将缓冲I/O内容清理掉,所以不会被打印出来

exit函数与_exit函数的区别

这里写图片描述
代码:

#include<stdio.h>
#include<stdlib.h>
#include<unistd.h>
int main()
{
  pid_t pid;
  pid = fork();
  if(pid<0)
  {
    perror("fork");
    exit(1);
  }
  else if(pid==0)
  {
    printf("this is exit process.\n");
    printf("exit test");
    exit(0);
  }
  else
  {
    printf("this is _exit process.\n");
    printf("_exit test");
   _exit(0);
  }
}

运行结果:

-bash-4.1$ ./a.out 
this is _exit process.
-bash-4.1$ this is exit process.
exit test

由此可看出,exit()会在进程终止前,将缓冲I/O内容清理掉,所以即使printf()函数里没有”\n”也会printf()的内容,而_exit()函数是直接终止进程,并未将缓冲I/O内容清理掉,所以不会被打印出来

exit()函数和_exit()函数最大的区别:

exit()函数在调用之前,会检查文件的打开情况,把文件缓冲区的内容写会文件,
而_exit()函数则不会进行任何操作,只是单纯的退出进程。

ps:exit()最后也会调用_exit(),但是调用exit()之前,还做了其他工作:
(1)执行用户通过atexit或on_exit定义的清理函数;
(2)关闭所有打开的流,所有的缓存数据均被写入;
(3)调用_exit().

return退出

return是一种更为常见的退出进程方法,执行return n等同于执行exit(n),
因为调用main的运行时函数会将main的返回值当做exit的参数

ps: printf函数使用的是缓冲I/O方式,该函数在遇到”\n”换行符时自动从缓冲区中将记录读出

在 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()` 是“立即退出”,适合子进程,避免缓冲区重复刷新。 --- ##
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值