免费注册 查看新帖 |

Chinaunix

  平台 论坛 博客 文库
12下一页
最近访问板块 发新帖
查看: 8772 | 回复: 10
打印 上一主题 下一主题

[Linux] 关于fork之后父子进程的文件描述符关系有些疑问? [复制链接]

论坛徽章:
0
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2015-01-14 02:22 |只看该作者 |倒序浏览

在重定向父进程的标准输出时,子进程的标准输出也被重定向。实际上,fork的一个特性是父进程的所有打开文件描述符都被复制到子进程中。父子进程的每个相同的打开描述符共享一个文件表项。
有点不理解的是,如果在fork之后父进程改变了文件描述符fd的指向,为什么子进程对应的文件描述符也会改变指向?

论坛徽章:
13
双鱼座
日期:2013-10-23 09:30:05数据库技术版块每日发帖之星
日期:2016-04-20 06:20:00程序设计版块每日发帖之星
日期:2016-03-09 06:20:002015亚冠之塔什干火车头
日期:2015-11-02 10:07:452015亚冠之德黑兰石油
日期:2015-08-30 10:07:07数据库技术版块每日发帖之星
日期:2015-08-28 06:20:00数据库技术版块每日发帖之星
日期:2015-08-05 06:20:002015年迎新春徽章
日期:2015-03-04 09:57:09辰龙
日期:2014-12-03 14:45:52酉鸡
日期:2014-07-23 09:46:23亥猪
日期:2014-03-13 08:46:22金牛座
日期:2014-02-11 09:36:21
2 [报告]
发表于 2015-01-14 09:41 |只看该作者
你上面不是说共享了嘛

论坛徽章:
46
2015小元宵徽章
日期:2015-03-06 15:58:18羊年新春福章
日期:2015-04-14 10:37:422015年亚洲杯之阿曼
日期:2015-04-14 10:41:50NBA常规赛纪念章
日期:2015-05-04 22:32:03NBA季后赛大富翁
日期:2015-05-04 22:34:11菠菜明灯
日期:2015-05-04 22:35:49新奥尔良黄蜂
日期:2015-05-04 22:49:2315-16赛季CBA联赛之广夏
日期:2015-12-11 15:02:342015年亚洲杯之巴勒斯坦
日期:2015-03-04 19:56:562015年亚洲杯之阿联酋
日期:2015-03-04 11:19:04休斯顿火箭
日期:2015-03-02 16:32:11纽约尼克斯
日期:2015-03-02 16:09:04
3 [报告]
发表于 2015-01-14 10:01 |只看该作者
上代码吧!!

论坛徽章:
0
4 [报告]
发表于 2015-01-14 18:41 |只看该作者
回复 2# bikong0411

恩恩,自己写了代码测试了一下,如果在fork之前改变描述符的指向,子进程与父进程的写到同一文件。也就是父子进程的每个相同的打开描述符共享一个文件表项。
但是如果在fork之后改变描述符的指向,则只会对父子进程中的一个产生影响。


   

论坛徽章:
0
5 [报告]
发表于 2015-01-14 18:45 |只看该作者
回复 3# super皮波
写了代码测试了一下,这是fork之后把标准输出重定向到一个hello文件
  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #include <unistd.h>
  4. #include <sys/stat.h>
  5. #include <sys/types.h>
  6. #include <fcntl.h>

  7. int main(void)
  8. {
  9.     pid_t pid;
  10.     int fd1, i = 1;

  11.     pid = fork();
  12.     if(pid < 0)
  13.     {
  14.         perror("fork");
  15.         exit(1);
  16.     }
  17.     else if(pid == 0)
  18.     {
  19.         while(1)
  20.         {
  21.             printf("write %d by child\n",i++);
  22.             sleep(1);
  23.         }
  24.     }
  25.     else
  26.     {
  27.         sleep(10);
  28.         fd1 = open("hello",O_CREAT | O_RDONLY | O_WRONLY | O_APPEND,0755);
  29.         if(fd1 < 0)
  30.         {
  31.             perror("open");
  32.             exit(1);
  33.         }
  34.         dup2(fd1,STDOUT_FILENO);
  35.         while(1)
  36.         {
  37.             printf("write %d by parent\n",i++);
  38.         }
  39.     }

  40.     return 0;
  41. }
复制代码
这个是在fork之前把标准输出重定向到hello文件
  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #include <unistd.h>
  4. #include <sys/stat.h>
  5. #include <sys/types.h>
  6. #include <fcntl.h>

  7. int main(void)
  8. {
  9.     pid_t pid;
  10.     int fd1, i = 1;

  11.     fd1 = open("hello",O_CREAT | O_RDONLY | O_WRONLY | O_APPEND,0755);
  12.     if(fd1 < 0)
  13.     {
  14.         perror("open");
  15.         exit(1);
  16.     }
  17.     dup2(fd1,STDOUT_FILENO);
  18.     pid = fork();
  19.     if(pid < 0)
  20.     {
  21.         perror("fork");
  22.         exit(1);
  23.     }
  24.     else if(pid == 0)
  25.     {
  26.         while(1)
  27.         {
  28.             printf("write %d by child\n",i++);
  29.             usleep(10000);
  30.         }
  31.     }
  32.     else
  33.     {
  34.         sleep(5);
  35.         while(1)
  36.         {
  37.             printf("write %d by parent\n",i++);
  38.             usleep(10000);
  39.         }
  40.     }

  41.     return 0;
  42. }
复制代码
刚开始的时候一直不太理解

论坛徽章:
46
2015小元宵徽章
日期:2015-03-06 15:58:18羊年新春福章
日期:2015-04-14 10:37:422015年亚洲杯之阿曼
日期:2015-04-14 10:41:50NBA常规赛纪念章
日期:2015-05-04 22:32:03NBA季后赛大富翁
日期:2015-05-04 22:34:11菠菜明灯
日期:2015-05-04 22:35:49新奥尔良黄蜂
日期:2015-05-04 22:49:2315-16赛季CBA联赛之广夏
日期:2015-12-11 15:02:342015年亚洲杯之巴勒斯坦
日期:2015-03-04 19:56:562015年亚洲杯之阿联酋
日期:2015-03-04 11:19:04休斯顿火箭
日期:2015-03-02 16:32:11纽约尼克斯
日期:2015-03-02 16:09:04
6 [报告]
发表于 2015-01-14 19:02 |只看该作者
回复 5# sagarfan
直接给你贴一张图


   

论坛徽章:
46
2015小元宵徽章
日期:2015-03-06 15:58:18羊年新春福章
日期:2015-04-14 10:37:422015年亚洲杯之阿曼
日期:2015-04-14 10:41:50NBA常规赛纪念章
日期:2015-05-04 22:32:03NBA季后赛大富翁
日期:2015-05-04 22:34:11菠菜明灯
日期:2015-05-04 22:35:49新奥尔良黄蜂
日期:2015-05-04 22:49:2315-16赛季CBA联赛之广夏
日期:2015-12-11 15:02:342015年亚洲杯之巴勒斯坦
日期:2015-03-04 19:56:562015年亚洲杯之阿联酋
日期:2015-03-04 11:19:04休斯顿火箭
日期:2015-03-02 16:32:11纽约尼克斯
日期:2015-03-02 16:09:04
7 [报告]
发表于 2015-01-14 19:08 |只看该作者
进程A的fd0和fd20这种就是dup类系统调用的结果
进程A的fd2和进程B的fd2就是fork的结果(fork之后指向相同的文件)
进程A的fd0和进程B的fd3最终指向同一个inode,这是这两个进程都调用过open的结果,此时两个文件不共享文件内的偏移
文件偏移是存放在第二个表中的,所以不管是dup还是fork,都是共享同一组偏移

论坛徽章:
46
2015小元宵徽章
日期:2015-03-06 15:58:18羊年新春福章
日期:2015-04-14 10:37:422015年亚洲杯之阿曼
日期:2015-04-14 10:41:50NBA常规赛纪念章
日期:2015-05-04 22:32:03NBA季后赛大富翁
日期:2015-05-04 22:34:11菠菜明灯
日期:2015-05-04 22:35:49新奥尔良黄蜂
日期:2015-05-04 22:49:2315-16赛季CBA联赛之广夏
日期:2015-12-11 15:02:342015年亚洲杯之巴勒斯坦
日期:2015-03-04 19:56:562015年亚洲杯之阿联酋
日期:2015-03-04 11:19:04休斯顿火箭
日期:2015-03-02 16:32:11纽约尼克斯
日期:2015-03-02 16:09:04
8 [报告]
发表于 2015-01-14 19:09 |只看该作者
还需要注意的是左边的是进程级别的,中间的是系统级别的,记录整个系统中打开的文件,inode可以理解成存在硬盘上的文件

论坛徽章:
13
双鱼座
日期:2013-10-23 09:30:05数据库技术版块每日发帖之星
日期:2016-04-20 06:20:00程序设计版块每日发帖之星
日期:2016-03-09 06:20:002015亚冠之塔什干火车头
日期:2015-11-02 10:07:452015亚冠之德黑兰石油
日期:2015-08-30 10:07:07数据库技术版块每日发帖之星
日期:2015-08-28 06:20:00数据库技术版块每日发帖之星
日期:2015-08-05 06:20:002015年迎新春徽章
日期:2015-03-04 09:57:09辰龙
日期:2014-12-03 14:45:52酉鸡
日期:2014-07-23 09:46:23亥猪
日期:2014-03-13 08:46:22金牛座
日期:2014-02-11 09:36:21
9 [报告]
发表于 2015-01-15 09:46 |只看该作者
回复 4# sagarfan


    子进程复制的父进程的文件描述符

论坛徽章:
0
10 [报告]
发表于 2015-01-15 13:49 |只看该作者
回复 7# super皮波
恩恩,非常感谢,但是那个dup产生的应该是进程A的fd1和fd20吧


   
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

北京盛拓优讯信息技术有限公司. 版权所有 京ICP备16024965号-6 北京市公安局海淀分局网监中心备案编号:11010802020122 niuxiaotong@pcpop.com 17352615567
未成年举报专区
中国互联网协会会员  联系我们:huangweiwei@itpub.net
感谢所有关心和支持过ChinaUnix的朋友们 转载本站内容请注明原作者名及出处

清除 Cookies - ChinaUnix - Archiver - WAP - TOP