免费注册 查看新帖 |

Chinaunix

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

问一个关于fork后的文件写入问题 [复制链接]

论坛徽章:
0
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2011-08-19 17:00 |只看该作者 |倒序浏览
代码如下(一个父进程,fork了100个子进程,每个子进程共享父进程打开的文件描述符,每个子进程写入1000个字符,基本的逻辑过程就是这样)

为什么在程序exit后,文件中,没有存入100*1000个字符?

如果将子进程数量增多,或者字符数增加,那么,文件中的字符数就更匹配不上了

很纳闷这个问题,坛子里的朋友有兴趣可以看一看,呵呵
  1. #include <stdio.h>
  2. #include <fcntl.h>

  3. int main()
  4. {
  5. struct flock stF;
  6. int i,n,fd,pid;

  7. fd=open("./output.txt",O_RDWR|O_CREAT);

  8. for(i=0;i<99;i++)
  9. {
  10.         pid=fork();

  11.         if(pid==-1)
  12.         {
  13.                 perror("fork:");
  14.         }
  15.         else if( pid > 0 )
  16.         {
  17.                 break;
  18.         }
  19. }

  20. stF.l_type=F_WRLCK;
  21. stF.l_start=0;
  22. stF.l_whence=SEEK_SET;
  23. stF.l_len=0;

  24. for(i=0;i<999;i++)
  25. {
  26.         if(fcntl(fd,F_SETLKW,&stF)==-1)
  27.         {
  28.                 perror("lock:");
  29.         }

  30.         lseek(fd,0,SEEK_END);
  31.         if(write(fd,"1",1)!=1)
  32.         {
  33.                 perror("write:");
  34.         }

  35.         stF.l_type=F_UNLCK;
  36.         if(fcntl(fd,F_SETLKW,&stF)==-1)
  37.         {
  38.                 perror("unlock:");
  39.         }
  40. }

  41. close(fd);
  42. }
复制代码

论坛徽章:
11
未羊
日期:2013-12-16 12:45:4615-16赛季CBA联赛之青岛
日期:2016-04-11 19:17:4715-16赛季CBA联赛之广夏
日期:2016-04-06 16:34:012015亚冠之卡尔希纳萨夫
日期:2015-11-10 10:04:522015亚冠之大阪钢巴
日期:2015-07-30 18:29:402015亚冠之城南
日期:2015-06-15 17:56:392015亚冠之卡尔希纳萨夫
日期:2015-05-15 15:19:272015亚冠之山东鲁能
日期:2015-05-14 12:38:13金牛座
日期:2014-12-04 15:34:06子鼠
日期:2014-10-16 13:40:4715-16赛季CBA联赛之八一
日期:2016-07-22 09:41:40
2 [报告]
发表于 2011-08-19 17:08 |只看该作者
至少有这个原因: i=0;i<999;i++

论坛徽章:
11
未羊
日期:2013-12-16 12:45:4615-16赛季CBA联赛之青岛
日期:2016-04-11 19:17:4715-16赛季CBA联赛之广夏
日期:2016-04-06 16:34:012015亚冠之卡尔希纳萨夫
日期:2015-11-10 10:04:522015亚冠之大阪钢巴
日期:2015-07-30 18:29:402015亚冠之城南
日期:2015-06-15 17:56:392015亚冠之卡尔希纳萨夫
日期:2015-05-15 15:19:272015亚冠之山东鲁能
日期:2015-05-14 12:38:13金牛座
日期:2014-12-04 15:34:06子鼠
日期:2014-10-16 13:40:4715-16赛季CBA联赛之八一
日期:2016-07-22 09:41:40
3 [报告]
发表于 2011-08-19 17:09 |只看该作者
还有这个原因: i=0;i<99;i++

论坛徽章:
0
4 [报告]
发表于 2011-08-19 17:21 |只看该作者
呵呵,循环的计数器引起的偏差,可以忽略不计,

如果将每个循环的计数条件改的小一些,例如10:10,或者100:100,都不会有问题

论坛徽章:
11
未羊
日期:2013-12-16 12:45:4615-16赛季CBA联赛之青岛
日期:2016-04-11 19:17:4715-16赛季CBA联赛之广夏
日期:2016-04-06 16:34:012015亚冠之卡尔希纳萨夫
日期:2015-11-10 10:04:522015亚冠之大阪钢巴
日期:2015-07-30 18:29:402015亚冠之城南
日期:2015-06-15 17:56:392015亚冠之卡尔希纳萨夫
日期:2015-05-15 15:19:272015亚冠之山东鲁能
日期:2015-05-14 12:38:13金牛座
日期:2014-12-04 15:34:06子鼠
日期:2014-10-16 13:40:4715-16赛季CBA联赛之八一
日期:2016-07-22 09:41:40
5 [报告]
发表于 2011-08-19 17:24 |只看该作者
本帖最后由 zylthinking 于 2011-08-19 17:33 编辑
代码如下(一个父进程,fork了100个子进程,每个子进程共享父进程打开的文件描述符,每个子进程写入1000个字 ...
liuzhuan23 发表于 2011-08-19 17:00


你要是没问题, 我佩服你。
你知道为什么我指出 i = 0; i < 99; i++么, 莫非我看不出来 1 + 99 == 100?
偶, 我看错了, 没看到break
但如果改小了就没有误差了, 我还是佩服你

论坛徽章:
11
未羊
日期:2013-12-16 12:45:4615-16赛季CBA联赛之青岛
日期:2016-04-11 19:17:4715-16赛季CBA联赛之广夏
日期:2016-04-06 16:34:012015亚冠之卡尔希纳萨夫
日期:2015-11-10 10:04:522015亚冠之大阪钢巴
日期:2015-07-30 18:29:402015亚冠之城南
日期:2015-06-15 17:56:392015亚冠之卡尔希纳萨夫
日期:2015-05-15 15:19:272015亚冠之山东鲁能
日期:2015-05-14 12:38:13金牛座
日期:2014-12-04 15:34:06子鼠
日期:2014-10-16 13:40:4715-16赛季CBA联赛之八一
日期:2016-07-22 09:41:40
6 [报告]
发表于 2011-08-19 17:51 |只看该作者
别的看不出来了

论坛徽章:
0
7 [报告]
发表于 2011-08-19 21:32 |只看该作者
呵呵,我说了,循环计数器差那么1-2个,无所谓的事情,根本就不是那的问题

不管是99,还是100,不可能会对写入的字符总数量造成那么大的影响

论坛徽章:
0
8 [报告]
发表于 2011-08-19 21:33 |只看该作者
有兴趣的话,可以贴到vi里面,试一试,cc一下,看看结果,很有意思

论坛徽章:
0
9 [报告]
发表于 2011-08-19 23:00 |只看该作者
本帖最后由 雨过白鹭洲 于 2011-08-19 23:01 编辑

楼上几位说的99和999只是代码逻辑问题,其实不是这个程序真正的问题

你运行程序100遍,output.txt文件的大小就会有100次不一样。

---------------------------------------------
很简单,问题在于进程调度,多进程写入同一个文件时存在竞争条件,而lseek和write组合在一起不是原子操作

问题代码:
lseek(fd,0,SEEK_END); // 如果系统刚好把进程挂起在这里,下次进程调度重新运行时将覆盖掉其它进程在此期间向文件写入的数据
if(write(fd,"1",1)!=1)


解决办法很简单,不用lseek调用,直接
fd = open("./output.txt", O_RDWR | O_CREAT | O_APPEND)

论坛徽章:
0
10 [报告]
发表于 2011-08-21 11:30 |只看该作者
另外你上面使用的锁,为什么仍然不能保证每次写都成功写到文件末尾,而不会覆盖其它进程写入文件的数据?

会不会跟内核I/O缓存有关。。
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

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

清除 Cookies - ChinaUnix - Archiver - WAP - TOP