(二十二)进程——进程原语fork

来源:互联网 发布:ubuntu如何卸载程序 编辑:程序博客网 时间:2024/04/29 05:32


  fork的作用是根据一个现有的进程复制出一个新进程,原来的进程称为父进程(Parent Process),新进程称为子进程(ChildProcess)。而一个进程包括代码、数据和分配给进程的资源,所以fork()函数通过系统调用创建一个与原来进程几乎完全相同的进程,一个进程调用fork()函数后,系统先给新的进程分配资源,包括存储数据和代码的空间。然后把原来的进程的所有值都复制到新的新进程中。相当于克隆了一个自己。

#include <unistd.h>pid_t fork(void);返回值:1)在父进程中,fork返回新创建子进程的进程ID;2)在子进程中,fork返回03)如果出现错误,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 testI am parent n = 10 , &n = 0xbf851d48I am child  n = 0 ,  &n = 0xbf851d48I am parent n = 10 , &n = 0xbf851d48I am child  n = 0 ,  &n = 0xbf851d48I am parent n = 10 , &n = 0xbf851d48I am child  n = 0 ,  &n = 0xbf851d48I am parent n = 10 , &n = 0xbf851d48I am child  n = 0 ,  &n = 0xbf851d48
5 0