1.exit()和_exit()
进程一般有两种退出方式,一种是异常终止,如调用abort(),另一种是使用exit()和_exit()系统调用正常终止。
![](https://images2018.cnblogs.com/blog/1435096/201807/1435096-20180722223435245-1566831687.png)
由上图片可知,exit与_exit区别:
1)exit会执行调用退出处理程序
2)exit会执行清除I/O缓存
接下来分别对于上述两个区别加以分析。
①区别一
首先,说一下退出处理程序产生的原因:以一个应用程序库为例,如果进程使用了该程序库,那么在进程终止前该库需要执行一些清理操作,由于库本身无法要求在进程退出前调用库中特定的清理函数,所以可以使用退出处理程序。退出处理程序是由程序员设计的函数,在进程调用exit()函数正常终止时自动执行。
可以使用atexit()函数注册退出处理程序。终止处理程序的调用与注册次序相反。
int atexit(void (*function)(void)) |
|
返回值 |
若成功则为0,若出错则为非零 |
void bye1(void)
{
printf("bye1...do\n");
}
void bye2(void)
{
printf("bye2...do\n");
}
int main()
{
atexit(bye1);
atexit(bye2);
printf("hello....");
exit(0);
}
②区别二
int main(void)
{
printf("hello。。。");
//exit(0);
fflush(stdout); //fflush(NULL);
_exit(0); //若没有fflush,打印不出东西
}
2.wait和waitpid
当子进程退出的时候,内核会向父进程发送SIGCHLD信号,子进程的退出是个异步事件(子进程可以在父进程运行的任何时刻终止)。
子进程退出时,内核将子进程置为僵尸状态,这个进程称为僵尸进程,它只保留最小的一些内核数据结构,以便父进程使用wait/waitpid查询子进程的退出状态。
pid_t wait(int *status) |
|
参数 |
状态信息 |
返回值 |
若成功则为子进程的ID,若出错则为-1 |
wait系统调用会使父进程暂停执行,直到它的一个子进程结束为止。
pid_t waitpid(pid_t pid, int *status,int options) |
|
作用 |
用来等待某个特定进程的结束 |
返回值 |
若成功则为子进程的ID,若出错则为-1 |
参数 |
|
pid |
pid == -1:等待任一子进程 |
|
pid > 0:等待其进程ID与pid相等的子进程 |
|
pid == 0: 等待与调用者进程同在一个组的进程 |
|
pid < -1: 等待其组ID等于pid的绝对值的任一子进程 |
status |
状态信息 |
options |
改变waitpid的行为,WNOHANG:防止waitpid把调用者的执行挂起 |
3.小demo
void TestFunc(int loopnum)
{
printf("loopnum:%d\n", loopnum);
}
int main(void )
{
int procnum = 10;
int loopnum = 100;
int i = 0, j = 0;
printf("please enter you procNum : \n");
scanf("%d", &procnum);
printf("please enter you loopnum :\n");
scanf("%d", &loopnum);
pid_t pid;
for (i=0; i<procnum; i++)
{
pid = fork();
if (pid == 0)
{
for (j=0; j<loopnum; j ++)
{
TestFunc(j);
}
exit(0);
}
}
//父进程能监控所有子进程的退出,避免僵尸进程
int mypid;
while ((mypdi=waitpid(-1,NULL,WNOHANG)) > 0)
{
;
}
printf("hello...\n");
return 0;
}
以上演示了父进程能监控所有子进程的退出,避免僵尸进程。