免费注册 查看新帖 |

Chinaunix

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

[C] 关于Linux下的 O_NONBLOCK 问题 [复制链接]

论坛徽章:
0
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2008-12-14 19:48 |只看该作者 |倒序浏览
APUE 高级IO一章中的一个示例,代码如下:

  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #include <unistd.h>
  4. #include <fcntl.h>
  5. #include <errno.h>

  6. int set_nonblock_flag(int fd, int value)
  7. {
  8.         int oldflag = fcntl(fd, F_GETFL, 0);
  9.         if(oldflag == -1)
  10.                 return -1;
  11.         
  12.         if(value != 0)
  13.                 oldflag |= O_NONBLOCK;
  14.         else
  15.                 oldflag &= ~O_NONBLOCK;
  16.         
  17.         return fcntl(fd, F_SETFL, oldflag);
  18.         
  19. }

  20. char buf[1000000];

  21. int main(void)
  22. {
  23.         int ntowrite, nwrite;
  24.         char *ptr;
  25.         
  26.         ntowrite = read(STDIN_FILENO, buf, sizeof(buf));
  27.         fprintf(stderr, "read %d bytes\n", ntowrite);
  28.         
  29.         set_nonblock_flag(STDOUT_FILENO, 1);
  30.         
  31.         ptr = buf;
  32.         while(ntowrite > 0)
  33.         {
  34.                 errno = 0;
  35.                 nwrite = write(STDOUT_FILENO, ptr, ntowrite);
  36.                 fprintf(stderr, "nwrite = %d, errno = %d\n", nwrite, errno);
  37.                
  38.                 if(nwrite > 0)
  39.                 {
  40.                         ptr += nwrite;
  41.                         ntowrite -= nwrite;
  42.                 }
  43.         }
  44.         
  45.         set_nonblock_flag(STDOUT_FILENO, 0);
  46.         
  47.         exit(0);
  48. }
复制代码


按照 ./a.out < testfile 2>stderr.out 这样的方式执行时(testfile的大小为6000000),发现结果与书上说的不一样。屏幕不停的输出,但stderr.out文件里并没有与非阻塞相关的输出,只有一句 read 6000000 bytes。感觉O_NONBLOCK在Linux下似乎无效,哪位高手能否给解释一下。

论坛徽章:
0
2 [报告]
发表于 2008-12-15 00:28 |只看该作者
哪位能在2.6.25版本(我这边是Fedora9)下试一下,告诉我stderr.out文件的内容。
谢谢了。

论坛徽章:
0
3 [报告]
发表于 2009-02-06 17:21 |只看该作者
书上有下面这段话:
The amount of data accepted by the terminal driver varies from system to system. The results will also vary depending on how you are logged in to the system: on the system console, on a hardwired terminal, on network connection using a pseudo terminal. If you are running a windowing system on your terminal, you are also going through a pseudo-terminal device.

书上的这段解释我看的不太懂

我在图形界面下,桌面上右键新建终端,打开一个终端,然后测试的
和楼主的测试结果一样,为什么会这样呢

论坛徽章:
0
4 [报告]
发表于 2009-02-06 17:24 |只看该作者
书上的给的一种测试结果如下
$ ls -l /etc/termcap                           print file size
-rw-r--r-- 1 root      702559 Feb 23  2002 /etc/termcap
$ ./a.out < /etc/termcap > temp.file           try a regular file first
read 500000 bytes
nwrite = 500000, errno = 0                     a single write
$ ls -l temp.file                              verify size of output file
-rw-rw-r-- 1 sar       500000 Jul   8 04:19 temp.file

为什么原来的文件/etc/termcap 是 702559 字节,为什么最终只读取500000字节呢
我测试的结果也是这样,不知道5000000是怎么来的

程序我没看仔细,原来数据是先读到buf里的,buf的大小是500000

[ 本帖最后由 zhuhefang2006 于 2009-2-6 18:48 编辑 ]

论坛徽章:
0
5 [报告]
发表于 2009-02-06 20:11 |只看该作者
现在运行又程序,和书上的运行结果一样了

还原不出楼主说的那种情况了

真是奇怪的
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

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

清除 Cookies - ChinaUnix - Archiver - WAP - TOP