【Linux 进程】fork父子进程间共享数据分析

之前我们通过fork()函数,得知了父子进程之间的存在着代码的拷贝,且父子进程都相互独立执行,那么父子进程是否共享同一段数据,即是否存在着数据共享。接下来我们就来分析分析父子进程是否存在着数据共享。

我们都知道,在linux下,内存存储的位置是全局变量,栈区,堆区,以及文件。字符常量区我们这里不作分析。下面我们依次以实际代码来验证它们是否存在着数据共享。所谓的父子进程数据共享,通俗点说就是父进程或者子进程对于数据的更改,会使得子进程或者父进程的数据同步更改。代码检测的思想是让父子进程中的一个修改数据,未对数据修改的进程调用数据,查看是否数据被修改,如果数据被修改,那么证明两者之间存在着数据共享,反之没有。

1、全局变量

#include<unistd.h>
#include<string.h>

int a=0;
int main()
{
      pid_t res=fork();
      if(res==0)
      {
            a=10;
            printf("child is %d\n",a);
      }
      else
      {
            sleep(2); //睡眠是为了保证子进程在父进程之前先对数据进行了修改
            printf("father is %d\n",a);
      }
      return 0;
}

通过结果得知,子进程对数据a进行了修改,但是父进程获取的数据确仍然是初始化的值。所以我们可以得知,在数据类型为全局变量时,父子进程之间的数据不共享。

 

2.栈区(局部变量区)

实现的代码与全局相似,只需要将int a=0在主函数中声明即可。这里直接给运行结果,不在展示源码。

代码运行结果如下:

我们可以看到,运行的结果与全局变量得到的结果一样,所以可以得出的结论是,当数据类型为局部变量的时候,父子进程之间的数据不共享。

3.堆区(动态内存)

 

可以看出,同样未有改变,与局部和全局的结果一致。则得出的结论是:当数据类型是动态开辟时,父子进程的数据不共享。

4.文件

在文件a.txt中存放了字符串hello world,然后父子进程分别去读文件,发现文件描述符偏移了2次。

通过结果我们可以看出,对于数据类型为文件时,父子进程之间共享数据,具体而言是共享了文件偏移量。

对此将上述进行整理后得到的父子进程之间数据共享结果如下:

全局变量           栈区(局部变量)           堆区(动态开辟)            文件

  不共享                      不共享                    不共享           共享文件偏移量    

文章转自:http://blog.sina.com.cn/s/blog_179ea19500102wz0x.html

转载:【Linux 进程】fork父子进程间共享数据分析 - 我得去图书馆了 - 博客园

  • 6
    点赞
  • 12
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: Linux中的fork()函数可以创建一个新的进程,这个新进程是原进程的副本,也就是说,它们有相同的代码、数据和堆栈。fork()函数会返回两次,一次是在进程中返回子进程进程ID,另一次是在子进程中返回。进程子进程的区别在于它们的进程ID不同,以及它们的进程ID也不同。进程子进程共享文件描述符、信号处理程序和文件锁等资源,但是它们各自拥有自己的地址空和堆栈。 ### 回答2: Linuxfork()是创建进程的一个系统调用,fork()会复制一份进程的全部资源,包括代码段、数据段、堆栈、打开的文件、进程组信息等都会被复制到子进程中。因此,原来进程有的资源,在子进程中都会有一个副本。fork()调用成功后,会返回两次,一次是在进程中返回子进程的PID,另一次是在子进程中返回0。 在fork()调用完成后,就会出现两个进程进程子进程进程中的所有资源都被完全复制到了子进程中,但子进程具有独立的内存空进程ID,因此进程子进程的内存空是互相独立的。在这种情况下,进程子进程执行的代码相同,但是子进程是一个全新的进程,具有自己的内存空进程上下文。 如果子进程想和进程进行通信,可以使用管道或者共享内存。透过管道或共享内存,子进程可以读取进程中的数据,或者将自己产生的数据发送给进程。另外,还可以使用信号或者消息队列进行通信。 需要特别注意的是,fork()调用成功后,子进程中会复制进程中的所有资源,包括打开的文件描述符等。因此,子进程需要关闭不需要的文件描述符,避免浪费系统资源。此外,子进程也需要确保在调用exec()函数之前,所有需要使用的文件描述符都已经打开了,否则在子进程中打开的文件描述符可能会覆盖进程中已经打开的文件描述符,导致出现错误。 ### 回答3: 在Linux系统中,每个进程都有一个唯一的进程ID(PID),以及一些其他的属性和信息。除了在启动时由init进程创建的特殊进程以外,每个进程都是由另一个进程fork”出来的,即在原有进程的基础上创建一个全新的进程。 在C语言中,可以使用fork()函数来实现这个操作。每次调用fork()函数时,会创建出一个全新的进程,称为“子进程”,并且这个子进程就是由“进程fork出来的。进程子进程在大部分方面都是相同的,例如二者运行相同的程序、拥有相同的内存空变量等等,但在某些方面也有一些差异,例如二者的进程ID不同(进程进程ID就是调用fork()函数前的进程ID,而子进程进程ID是新分配的),以及二者对共享资源的访问方式不同(具体取决于程序的实现方式)。 通常情况下,fork()函数的返回值为0,表示子进程;或者返回一个大于0的数值,表示进程,并且这个数值就是子进程进程ID。如果fork()函数返回一个负数,则表示创建子进程失败。 在实际编程中,可以利用fork()函数来实现一些复杂的应用,例如多进程并行计算、进程通信等等。此外,fork()函数也是Unix/Linux系统中一种重要的机制,可以实现进程的动态创建和销毁,从而增强了系统的灵活性和可扩展性。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值