免费注册 查看新帖 |

Chinaunix

  平台 论坛 博客 文库
最近访问板块 发新帖
楼主: liuzhuan23
打印 上一主题 下一主题

问一个关于fork后的文件操作的IO问题 [复制链接]

论坛徽章:
95
程序设计版块每日发帖之星
日期:2015-09-05 06:20:00程序设计版块每日发帖之星
日期:2015-09-17 06:20:00程序设计版块每日发帖之星
日期:2015-09-18 06:20:002015亚冠之阿尔艾因
日期:2015-09-18 10:35:08月度论坛发贴之星
日期:2015-09-30 22:25:002015亚冠之阿尔沙巴布
日期:2015-10-03 08:57:39程序设计版块每日发帖之星
日期:2015-10-05 06:20:00每日论坛发贴之星
日期:2015-10-05 06:20:002015年亚冠纪念徽章
日期:2015-10-06 10:06:482015亚冠之塔什干棉农
日期:2015-10-19 19:43:35程序设计版块每日发帖之星
日期:2015-10-21 06:20:00每日论坛发贴之星
日期:2015-09-14 06:20:00
1 [报告]
发表于 2011-08-19 23:57 |显示全部楼层
我寫了一個:

  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #include <unistd.h>
  4. #include <sys/stat.h>
  5. #include <fcntl.h>

  6. #define CNT 10

  7. void write_file(int fd, char data, size_t len);

  8. int main(int argc, char *argv[])
  9. {
  10.         char *fn;
  11.         int fd;
  12.         pid_t pid;

  13.         if (argc > 1)
  14.                 fn = argv[1];
  15.         else
  16.                 fn = "flock-demo";

  17.         if ((fd = open(fn, O_WRONLY | O_CREAT, S_IRUSR | S_IWUSR | S_IRGRP | S_IROTH)) < 0) {
  18.                 perror("open:");
  19.                 exit(-1);
  20.         }

  21.         for (int i = 0; i < CNT; i++) {
  22.                 if ((pid = fork()) < 0) {
  23.                         perror("fork:");
  24.                         exit(-1);
  25.                 } else {
  26.                         if (pid == 0) {
  27.                                 /* child process */
  28.                                 write_file(fd, i, sizeof(i));
  29.                                 exit(0);
  30.                         }
  31.                 }
  32.         }

  33.         return 0;
  34. }

  35. void write_file(int fd, char data, size_t len)
  36. {
  37.         struct flock fl;

  38.         /* lock the whole file */
  39.         fl.l_whence = SEEK_SET;
  40.         fl.l_start  = 0;
  41.         fl.l_len    = 0;

  42.         for (int i = 0; i < CNT; i++) {
  43.                 fl.l_type = F_WRLCK;
  44.                 if (fcntl(fd, F_SETLKW, &fl) < 0) {
  45.                         perror("fcntl:");
  46.                         exit(-1);
  47.                 }

  48.                 ssize_t wlen = write(fd, &data, len);
  49.                 if (wlen < 0) {
  50.                         perror("write:");
  51.                         exit(-1);
  52.                 } else {
  53.                         if ((size_t)wlen < len) {
  54.                                 fprintf(stderr, "Other write error in process %d\n", getpid());
  55.                                 exit(-1);
  56.                         }
  57.                 }

  58.                 fl.l_type = F_UNLCK;
  59.                 if (fcntl(fd, F_SETLKW, &fl) < 0) {
  60.                         perror("fcntl:");
  61.                         exit(-1);
  62.                 }
  63.         }
  64. }
复制代码
查看結果:

  1. lee@debian:~/tmp/C$ gcc -std=c99 flock.c
  2. lee@debian:~/tmp/C$ ./a.out
  3. lee@debian:~/tmp/C$ od -A d -w40 -t xI -v flock-demo
  4. 0000000 00000000 00000000 00000000 00000000 00000000 00000000 00000000 00000000 00000000 00000000
  5. 0000040 00000001 00000001 00000001 00000001 00000001 00000001 00000001 00000001 00000001 00000001
  6. 0000080 00000002 00000002 00000002 00000002 00000002 00000002 00000002 00000002 00000002 00000002
  7. 0000120 00000003 00000003 00000003 00000003 00000003 00000003 00000003 00000003 00000003 00000003
  8. 0000160 00000004 00000004 00000004 00000004 00000004 00000004 00000004 00000004 00000004 00000004
  9. 0000200 00000005 00000005 00000005 00000005 00000005 00000005 00000005 00000005 00000005 00000005
  10. 0000240 00000006 00000006 00000006 00000006 00000006 00000006 00000006 00000006 00000006 00000006
  11. 0000280 00000007 00000007 00000007 00000007 00000007 00000007 00000007 00000007 00000007 00000007
  12. 0000320 00000008 00000008 00000008 00000008 00000008 00000008 00000008 00000008 00000008 00000008
  13. 0000360 00000009 00000009 00000009 00000009 00000009 00000009 00000009 00000009 00000009 00000009
  14. 0000400
复制代码
若要各個子進程的輸出不互相穿插,則需要把加鎖放在子進程輸出循環的外面。

论坛徽章:
95
程序设计版块每日发帖之星
日期:2015-09-05 06:20:00程序设计版块每日发帖之星
日期:2015-09-17 06:20:00程序设计版块每日发帖之星
日期:2015-09-18 06:20:002015亚冠之阿尔艾因
日期:2015-09-18 10:35:08月度论坛发贴之星
日期:2015-09-30 22:25:002015亚冠之阿尔沙巴布
日期:2015-10-03 08:57:39程序设计版块每日发帖之星
日期:2015-10-05 06:20:00每日论坛发贴之星
日期:2015-10-05 06:20:002015年亚冠纪念徽章
日期:2015-10-06 10:06:482015亚冠之塔什干棉农
日期:2015-10-19 19:43:35程序设计版块每日发帖之星
日期:2015-10-21 06:20:00每日论坛发贴之星
日期:2015-09-14 06:20:00
2 [报告]
发表于 2011-08-21 23:38 |显示全部楼层
楼主的问题不是出在记录锁问题,而是子进程互相做了其他进程地址空间的副本,
并且在for....for()中 ...
hohoss 发表于 2011-08-21 15:28



    他代碼的主要問題還就是把鎖用錯了。在那個寫循環中,lock type 只在第一次執行時設成了 F_WRLCK, 後面就都是 F_UNLCK 了,相當於沒有鎖。

论坛徽章:
95
程序设计版块每日发帖之星
日期:2015-09-05 06:20:00程序设计版块每日发帖之星
日期:2015-09-17 06:20:00程序设计版块每日发帖之星
日期:2015-09-18 06:20:002015亚冠之阿尔艾因
日期:2015-09-18 10:35:08月度论坛发贴之星
日期:2015-09-30 22:25:002015亚冠之阿尔沙巴布
日期:2015-10-03 08:57:39程序设计版块每日发帖之星
日期:2015-10-05 06:20:00每日论坛发贴之星
日期:2015-10-05 06:20:002015年亚冠纪念徽章
日期:2015-10-06 10:06:482015亚冠之塔什干棉农
日期:2015-10-19 19:43:35程序设计版块每日发帖之星
日期:2015-10-21 06:20:00每日论坛发贴之星
日期:2015-09-14 06:20:00
3 [报告]
发表于 2011-08-23 13:14 |显示全部楼层
嗯,确实是锁没起作用,不过话说文件描述符都是继承下来的,都指向内核打开文件表中相同的文件, ...
skymoondyy 发表于 2011-08-23 11:16



    你这是假设 write 操作是原子的。若真是这样,这里的锁还真没必要。

论坛徽章:
95
程序设计版块每日发帖之星
日期:2015-09-05 06:20:00程序设计版块每日发帖之星
日期:2015-09-17 06:20:00程序设计版块每日发帖之星
日期:2015-09-18 06:20:002015亚冠之阿尔艾因
日期:2015-09-18 10:35:08月度论坛发贴之星
日期:2015-09-30 22:25:002015亚冠之阿尔沙巴布
日期:2015-10-03 08:57:39程序设计版块每日发帖之星
日期:2015-10-05 06:20:00每日论坛发贴之星
日期:2015-10-05 06:20:002015年亚冠纪念徽章
日期:2015-10-06 10:06:482015亚冠之塔什干棉农
日期:2015-10-19 19:43:35程序设计版块每日发帖之星
日期:2015-10-21 06:20:00每日论坛发贴之星
日期:2015-09-14 06:20:00
4 [报告]
发表于 2011-08-24 13:09 |显示全部楼层
lseek 还是需要的 虽然共享了文件大小

seek 是没有必要的。

不过确实不知道为什么会出现LZ的 ...
atsjun 发表于 2011-08-24 12:53


他把锁用错了,根本没锁上。

论坛徽章:
95
程序设计版块每日发帖之星
日期:2015-09-05 06:20:00程序设计版块每日发帖之星
日期:2015-09-17 06:20:00程序设计版块每日发帖之星
日期:2015-09-18 06:20:002015亚冠之阿尔艾因
日期:2015-09-18 10:35:08月度论坛发贴之星
日期:2015-09-30 22:25:002015亚冠之阿尔沙巴布
日期:2015-10-03 08:57:39程序设计版块每日发帖之星
日期:2015-10-05 06:20:00每日论坛发贴之星
日期:2015-10-05 06:20:002015年亚冠纪念徽章
日期:2015-10-06 10:06:482015亚冠之塔什干棉农
日期:2015-10-19 19:43:35程序设计版块每日发帖之星
日期:2015-10-21 06:20:00每日论坛发贴之星
日期:2015-09-14 06:20:00
5 [报告]
发表于 2011-08-27 00:22 |显示全部楼层
回复  hohoss


    fork虽然会暂时公用一些东西,但是有写时复制机制,本身就会避免两个进程间互相修改 ...
vincentff7 发表于 2011-08-26 21:58



    Linux kernel 的 file table 在各個進程間是共享的,而 file offset 恰好在 file table 中,見 APUE Section 3.10 File Sharing

论坛徽章:
95
程序设计版块每日发帖之星
日期:2015-09-05 06:20:00程序设计版块每日发帖之星
日期:2015-09-17 06:20:00程序设计版块每日发帖之星
日期:2015-09-18 06:20:002015亚冠之阿尔艾因
日期:2015-09-18 10:35:08月度论坛发贴之星
日期:2015-09-30 22:25:002015亚冠之阿尔沙巴布
日期:2015-10-03 08:57:39程序设计版块每日发帖之星
日期:2015-10-05 06:20:00每日论坛发贴之星
日期:2015-10-05 06:20:002015年亚冠纪念徽章
日期:2015-10-06 10:06:482015亚冠之塔什干棉农
日期:2015-10-19 19:43:35程序设计版块每日发帖之星
日期:2015-10-21 06:20:00每日论坛发贴之星
日期:2015-09-14 06:20:00
6 [报告]
发表于 2011-08-29 17:11 |显示全部楼层
seek没必要,因为父子进程共享文件的offset。write只写一个字节怎么也是原子操作了吧,所以按理锁也没有必要。所以我觉得这个问题,应该是多核处理器的问题。如果是单核,不会有这种问题。
skymoondyy 发表于 2011-08-29 09:48



    不要总是“觉得”、“按理”,测试一下嘛。
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

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

清除 Cookies - ChinaUnix - Archiver - WAP - TOP