exit()和_exit()的区别

本文详细介绍了Linux中进程退出的各种方式,包括正常退出与异常退出的区别,以及exit与_exit函数的使用和区别。并通过示例展示了这两种函数的行为差异。

进程就好比人一样有其生命,我们通过fork()函数来创建一个进程,那么我们又是如何来中止进程呢。

进程退出

1.在Linux中任何让一个进程退出

进程退出表示进程即将结束。在Linux中进程退出分为了正常退出和异常退出两种。

1>正常退出

a. 在main()函数中执行return 。


b.调用exit()函数

c.调用_exit()函数

2>异常退出

a.调用about函数

b.进程收到某个信号,而该信号使程序终止。


Tiger-John说明:
不管 是那种 退出方式,系统最终都会执行内核中的同一代码。这段代码用来关闭进程所用已打开的文件描述符,释放它所占用的内存和其他资源。





3>比较以上几种退出方式的不同点

(1)exit和return 的区别:

a.exit是一个函数,有参数。exit执行完后把控制权交给系统

b.return是函数执行完后的返回。renturn执行完后把控制权交给调用函数。

(2)exit和abort的区别:

a.exit是正常终止进程

b.about是异常终止。





现在我们重点了解exit()和_exit()函数



2.exit()和_exit()的学习

1>exit和_exit函数都是用来终止进程的。

当程序执行到exit或_exit时,系统无条件的停止剩下所有操作,清除包括PCB在内的各种数据结构,并终止本进程的运行。

2>exit在头文件stdlib.h中声明,而_exit()声明在头文件unistd.h中声明。 exit中的参数exit_code为0代表进程正常终止,若为其他值表示程序执行过程中有错误发生。

3>exit()和_exit()的区别:


a._exit()执行后立即返回给内核,而exit()要先执行一些清除操作,然后将控制权交给内核。

b. 调用_exit函数时,其会关闭进程所有的文件描述符,清理内存以及其他一些内核清理函数,但不会刷新流(stdin, stdout, stderr ...). exit函数是在_exit函数之上的一个封装,其会调用_exit,并在调用之前先刷新流。

Tiger-John说明:

exit()函数与_exit()函数最大区别就在于exit()函数在调用exit系统之前要检查文件的打开情况,把文件缓冲区的内容写回文件。由于Linux的标准函数库中,有一种被称作“缓冲I/O”的操作,其特征就是对应每一个打开的文件,在内存中都有一片缓冲区。每次读文件时,会连续的读出若干条记录,这样在下次读文件时就可以直接从内存的缓冲区读取;同样,每次写文件的时候也仅仅是写入内存的缓冲区,等满足了一定的条件(如达到了一定数量或遇到特定字符等),再将缓冲区中的内容一次性写入文件。这种技术大大增加了文件读写的速度,但也给编程代来了一点儿麻烦。比如有一些数据,认为已经写入了文件,实际上因为没有满足特定的条件,它们还只是保存在缓冲区内,这时用_exit()函数直接将进程关闭,缓冲区的数据就会丢失。因此,要想保证数据的完整性,就一定要使用exit()函数。

c . 通过一个函数实例来看看它们之间的区别:

函数实例1 : exit.c

1 #include<stdio.h>
2 #include<stdlib.h>
3
4 int main()
5 {
6 printf("using exit----\n");
7 printf("This is the content in buffer\n");
8 exit(0);
9 }

函数经过调试后

think@ubuntu:~/work/process_thread/exit$ gcc exit.c -o exit
think@ubuntu:~/work/process_thread/exit$ ./exit


执行结果为:

using exit----
This is the content in buffer

函数实例2:_exit.c

1 #include<stdio.h>
2 #include<unistd.h>
3
4 int main(void)
5 {
6 printf("using _exit--\n");
7 printf("This is the content in buffer");
8 _exit(0);
9 }
函数经过调试后

think@ubuntu:~/work/process_thread/exit$ gcc _exit.c -o _exit
think@ubuntu:~/work/process_thread/exit$ ./_exit


执行结果为 :

using _exit--


Tiger-John说明:

1.printf函数就是使用缓冲I/O的方式,该函数在遇到“\n”换行符时自动的从缓冲区中将记录读出。所以exit()将缓冲区的数据写完后才退出,而_exit()函数直接退出。

2.大家也可以把函数实例2中的printf("This is the content in buffer");改为printf("This is the content in buffer\n")(即在printf中最后加一个\n看运行结果是什么,为什么会产生这样的结果呢?)

Tiger-John补充:

父子进程终止的先后顺序不同会产生不同的结果。

1>父进程先于子进程终止:

此种情况就是我们前面所用的孤儿进程。当父进程先退出时,系统会让init进程接管子进程 。

2>子进程先于父进程终止,而父进程又没有调用wait函数

此种情况子进程进入僵死状态,并且会一直保持下去直到系统重启。子进程处于僵死状态时,内核只保存进程的一些必要信息以备父进程所需。此时子进程始终占有着资源,同时也减少了系统可以创建的最大进程数。

什么是 僵死状态呢?

一个已经终止、但是其父进程尚未对其进行善后处理(获取终止子进程的有关信息,释放它仍占有的资源)的进程被称为僵死进程(zombie)。ps命令将僵死进程的状态打印为Z

3> 子进程先于父进程终止,而父进程调用了wait函数

此时父进程会等待子进程结束。

在 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、付费专栏及课程。

余额充值