[转帖]linux中fork函数问题讲解_VMware, Unix及操作系统讨论区_Weblogic技术|Tuxedo技术|中间件技术|Oracle论坛|JAVA论坛|Linux/Unix技术|hadoop论坛_联动北方技术论坛  
网站首页 | 关于我们 | 服务中心 | 经验交流 | 公司荣誉 | 成功案例 | 合作伙伴 | 联系我们 |
联动北方-国内领先的云技术服务提供商
»  游客             当前位置:  论坛首页 »  自由讨论区 »  VMware, Unix及操作系统讨论区 »
总帖数
1
每页帖数
101/1页1
返回列表
0
发起投票  发起投票 发新帖子
查看: 3684 | 回复: 0   主题: [转帖]linux中fork函数问题讲解        下一篇 
匿名用户
发表于: IP:您无权察看 2013-6-28 12:14:38 | [全部帖] [楼主帖] 楼主

今天偶然间发现一个问题,就是fork的时候莫民奇妙的,多出了不必要 进程,然后自己写了个小小测试程序,问题依旧,没研究明白,贴出来请高手指教!!!!小弟先谢谢了
测试代码:

void fun()
{
      static int g_var = 99;
      printf("ppid[%d] pid[%d] g_var[%d]\n",getppid(),getpid(),g_var);
      g_var += 10 ;
      printf("ppid[%d] pid[%d] g_var[%d]\n",getppid(),getpid(),g_var);
}
int main(int argc,char* argv[])
{
      int sock_fd1 = 0,sock_fd2 = 0;
      printf("pid[%d]\n",getpid());
      sock_fd1 = fork();
      if(sock_fd1 < 0)
      {
            printf("sock_fd1 < 0\n");
            return -1;
      }
      else if(sock_fd1 == 0)
      {
            fun();
            close(sock_fd1);
      }
      else
      {
            close(sock_fd1);
      }
      printf("-----------------------------\n");
      sock_fd2 = fork();
      if(sock_fd2 < 0)
      {
            printf("sock_fd2 < 0\n");
            return 0;
      }
      else if(sock_fd2 == 0)
      {
            fun();
            close(sock_fd2);
      }
      else
      {
            close(sock_fd2);
      }
      return 0;
}


结果:

pid[4135]
-----------------------------
ppid[4135] pid[4136] g_var[99]
ppid[1] pid[4136] g_var[109]
-----------------------------
[cgh@localhost test]$ ppid[1] pid[4137] g_var[99]
ppid[1] pid[4137] g_var[109]
ppid[1] pid[4138] g_var[109]
ppid[1] pid[4138] g_var[119]


问题:
1、为什么会有4个进程出现,实际fork出了3个进程?
2、为什么除了第一打印出ppid为4135,后面所有打印ppid都是1,即使是在第一次fork的进程中调用fun()时,第二次打印的ppid都变了??
3、根据打印的结果分析,又两个进程调用fun()时,g_var的初始值都是99,为什么会这样,静态变量不是存放在堆中的静态变量区么,每次调用后g_var的都该变化才对啊??

我提供的解释:

1)fork()出进程数的问题:
  fork出的进程类似于树的结构:进程总数n_pid = 2的(fork数m)次幂(举例:2次fork,总共产生4个进程)

2)第一次执行时可能父进程执行了 close(sock_fd1),这样父进程被强制关闭,至此所有子进程4136,4137,4138就变成了孤儿进程,Linux规定所有进程的祖先进程都为1,所以此时系统将这些孤儿进程都归在pid =1的祖先进程名下,所以你看到他们的父进程都变为1了

3)首先fork原理:复制出的每一个进程只是继承(复制)父进程的初始资源,之后自己操作自己的副本,互不牵连(例如:ppid =3345  static int a = 5;
                  子进程 pid =3346 他会复制 a = 5一个副本,之后他就操作这个副本,和父进程没有任何关系)

第一个fork():
产生了4416,此时g_var=99(父进程4415);执行一次它的g_var=109;
第二个fork():
4415创建一个4137,此时g_var=99(父进程4415);执行一次它的g_var=109
4416创建一个4138,此时g_var=109(这个子进程的父进程4416);执行一次它的g_var=119



赞(0)    操作        顶端 
总帖数
1
每页帖数
101/1页1
返回列表
发新帖子
请输入验证码: 点击刷新验证码
您需要登录后才可以回帖 登录 | 注册
技术讨论