fork的作用是根据一个现有的进程复制出一个新进程,原来的进程称为父进程(Parent Process),新进程称为子进程(ChildProcess)。而一个进程包括代码、数据和分配给进程的资源,所以fork()函数通过系统调用创建一个与原来进程几乎完全相同的进程,一个进程调用fork()函数后,系统先给新的进程分配资源,包括存储数据和代码的空间。然后把原来的进程的所有值都复制到新的新进程中。相当于克隆了一个自己。
#include <unistd.h>
pid_t fork(void);
返回值:
1)在父进程中,fork返回新创建子进程的进程ID;
2)在子进程中,fork返回0;
3)如果出现错误,fork返回一个负值;
·
·
·
子进程会复制父进程的0到3g空间和父进程内核中的PCB,但id号不同。
注:fork调用一次返回两次(子进程一次,父进程一次)
+ 父进程中返回子进程ID
+ 子进程中返回0
+ 读时共享,写时复制
接下来用一个例子说明:
(例子中牵涉到很多重要的问题,请认真阅读)
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
int main(void)
{
pid_t pid;
int n = 0;
printf("this is a test\n");
/*
*上面的语句如果没有\n那么就会在子进程中也打印一句
*这个是由于printf的机制造成的,遇到\n就出栈,
*若没有\n那么会在执行下一条语句前出栈,
*而下一条语句是fork,此时已经复制主进程的环境,其中包括打印
*/
pid = fork();//调用fork时,子进程会从下面的语句开始执行
if(pid > 0)
{
while(1)
{
n = 10;//两个n的值是不一样的,那是因为读时共享,写时复制
printf("I am parent n = %d , &n = %p\n",n,&n);
sleep(2);
}
}
else if(pid == 0)
{
while(1)
{
//两个n的地址是一样的,那是因为虚拟地址一样的原因
//但是n的值是不一样的,那是因为读时共享,写时复制
printf("I am child n = %d , &n = %p\n",n,&n);
sleep(2);
}
}
else
{
perror("fork");
exit(1);
}
return 0;
}
输出的结果是:
book@ubuntu:~/work/LinuxProgment/进程/3fork$ ./app
this is a test
I am parent n = 10 , &n = 0xbf851d48
I am child n = 0 , &n = 0xbf851d48
I am parent n = 10 , &n = 0xbf851d48
I am child n = 0 , &n = 0xbf851d48
I am parent n = 10 , &n = 0xbf851d48
I am child n = 0 , &n = 0xbf851d48
I am parent n = 10 , &n = 0xbf851d48
I am child n = 0 , &n = 0xbf851d48