进程结束
1.在Linux中任何让一个进程结束
进程退出表示进程即将结束。在Linux中进程退出分为了正常退出和异常退出两
种。
1>正常退出
a. 在main()函数中执行return 。
b.调用exit()函数
c.调用_exit()函数
2>异常退出
a.调用about函数
b.进程收到某个信号 ,而该信号使程序终止。
不管 是哪种退出方式 ,系统最终都会执行内核中的同一代码。这段代码用来关闭进
程所用已打开的文件描述符 ,释放它所占用的内存和其他资源。
3.比较以上几种退出方式的不同点
(1)exit和return 的区别 :
a.exit是一个函数 ,有参数。exit执行完后把控制权交给系统
b.return是函数执行完后的返回。renturn执行完后把控制权交给调用函数。
(2)exit和abort的区别 :
a.exit是正常终止进程
b.about是异常终止。
exit()和_exit()函数
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 ,并在调用之前先刷新流。
exit()函数与_exit()函数最大区别就在于exit()函数在调用exit系统之前要检查文件的
打开情况 ,把文件缓冲区的内容写回文件。由于Linux的标准函数库中,有一种被
称作“缓冲I/O”的操作 ,其特征就是对应每一个打开的文件 ,在内存中都有一片缓冲
区。每次读文件时 ,会连续的读出若干条记录 ,这样在下次读文件时就可以直接从
内存的缓冲区读取 ;同样 ,每次写文件的时候也仅仅是写入内存的缓冲区,等满足
了一定的条件 (如达到了一定数量或遇到特定字符等),再将缓冲区中的内容一次
性写入文件。这种技术大大增加了文件读写的速度 ,但也给编程代来了一点儿麻
烦。比如有一些数据 ,认为已经写入了文件 ,实际上因为没有满足特定的条件 ,它
们还只是保存在缓冲区内,这时用_exit()函数直接将进程关闭,缓冲区的数据就会
丢失。因此 ,要想保证数据的完整性 ,就一定要使用exit()函数。
几个要点 :
1.不管进程如何终止 ,最后都会执行内核中的同一段代码 :为相应进程关闭所有打
开描述符 ,释放内存等等。
2.若父进程在子进程之前终止了,则子进程的父进程将变为init进程 ,其PID为1 ;
保证每个进程都有父进程。
3.当子进程先终止 ,父进程如何知道子进程的终止状态?事实上 ,内核为每个终止
子进程保存了终止状态等信息 ,父进程调用wait等函数 ,就可获取该信息。
4.当父进程调用wait等函数后 ,内核将释放终止进程所使用的所有内存 ,关闭其打
开的所有文件。
5.对于已经终止、但是其父进程尚未对其调用wait等函数的进程 ,被称为僵尸进程
(即已经结束 ,但尚未释放资源的)。
6.对于父进程先终止 ,而被init领养的进程会是僵尸进程吗?init对每个终止的子进
程 ,都会调用wait函数 ,获取其终止状态信息。
综上所述 ,子进程调用exit后 ,父进程必须调用wait。
进程等待
系统中的僵尸进程都要由wait系统调用来回收,下面就通过实战看一看wait的具体用法 :
wait的函数原型是 :
#include /* 提供类型pid_t的定义 */
#include
pid_t wait(int *status);
进程一旦调用了wait ,就立即阻塞自己,由wait 自动分析是否当前进程的某个子进程已经退出,如果让它
找到了这样一个已经变成僵尸的子进程 , wait就会收集这个子进程的信息 ,并把它彻底销毁后返回;如
果没有找到这样一个子进程 ,wait就会一直阻塞在这里