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


  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 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
  • 6
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值