fork函数初识
pid_t fork(void ) ;
//返回值:子进程中返回0,父进程返回子进程id,出错返回 -1
进程调用fork,当控制转移到内核中的fork代码后,内核做:
分配新的内存块和内核数据结构给子进程
将父进程部分数据结构内容拷贝至子进程
添加子进程到系统进程列表当中
fork返回,开始调度器调度
#include<stdio.h>
#include<unistd.h>
int main()
{
int id=fork();
if(id==0)//子进程
{
printf("我是子进程,pid = %d,ppid = %d \n",getpid(),getppid());
sleep(1);
}
else
{
printf("我是父进程,pid = %d,ppid = %d \n",getpid(),getppid());
sleep(1);
}
return 0;
}
写时拷贝
通常,父子代码共享,父子再不写入时,数据也是共享的,当任意一方试图写入,便以写时拷贝的方式各自一份副本。具体见下图:
进程退出
常规的退出方法:
- 从main返回
- 调用exit
- ctrl + c,信号终止
int main ()
{
return 0;
}
int main()
{
exit(0);
}
进程等待
子进程退出,父进程如果不管不顾,就可能造成‘僵尸进程’的问题,进而造成内存泄漏。
另外,进程一旦变成僵尸状态,那就刀枪不入,“杀人不眨眼”的kill -9 也无能为力,因为谁也没有办法杀死一个已经死去的进程。
最后,父进程派给子进程的任务完成的如何,我们需要知道。如,子进程运行完成,结果对还是不对,或者是否正常退出。
父进程通过进程等待的方式,回收子进程资源,获取子进程退出信息
waitpid方法
pid_ t waitpid ( pid_t pid, int *status, int options ) ;
返回值:
当正常返回的时候waitpid返回收集到的子进程的进程ID;
如果设置了选项WNOHANG,而调用中waitpid发现没有已退出的子进程可收集,则返回0;
如果调用中出错,则返回 -1,这时errno会被设置成相应的值以指示错误所在;
参数:
pid: pid=-1,等待任一个子进程。与wait等效。
pid>0 .等待其进程ID与pid相等的子进程。
status : WIFEXITED( status ) : 若为正常终止子进程返回的状态,则为真。(查看进程是否是正常退出)
WEXITSTATUS ( status ) : 若WIFEXITED非零,提取子进程退出码。(查看进程的退出码)
options : WNOHANG : 若pid指定的子进程没有结束,则waitpid ()函数返回0,不予以等待。若正常结束,则返回该子进程的ID。