免费注册 查看新帖 |

Chinaunix

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

[C] 进程通信管道问题 [复制链接]

论坛徽章:
1
白羊座
日期:2013-10-17 23:11:46
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2013-02-28 21:50 |只看该作者 |倒序浏览
《unix环境高级编程》403页


明明是两个管道为什么read的时候会发生阻塞呢?请各位大侠指点迷津啊~

论坛徽章:
0
2 [报告]
发表于 2013-02-28 23:11 |只看该作者
新人,学习下

论坛徽章:
0
3 [报告]
发表于 2013-02-28 23:39 |只看该作者
本帖最后由 txgc_wm 于 2013-02-28 23:41 编辑

read依赖于管道的打开模式,打开管道时可使用pipe2设定相应的flags。书上所写的阻塞是在管道中没有数据的情况下发生的。

       int pipe2(int pipefd[2], int flags);

DESCRIPTION
       pipe()  creates  a  pipe, a unidirectional data channel that can be used for interprocess communication.  The array pipefd is used to return
       two file descriptors referring to the ends of the pipe.  pipefd[0] refers to the read end of the pipe.  pipefd[1] refers to the write end of
       the  pipe.  Data written to the write end of the pipe is buffered by the kernel until it is read from the read end of the pipe.  For further
       details, see pipe(7).

       If flags is 0, then pipe2() is the same as pipe().  The following values can be bitwise ORed in flags to obtain different behavior:

       O_NONBLOCK  Set the O_NONBLOCK file status flag on the two new open file descriptions.  Using this flag saves extra  calls  to  fcntl(2)  to
                   achieve the same result.

       O_CLOEXEC   Set  the  close-on-exec  (FD_CLOEXEC) flag on the two new file descriptors.  See the description of the same flag in open(2) for
                   reasons why this may be useful.

同时也可以查看内核文件fs/pipe.c中的pipe_read函数实现。

论坛徽章:
0
4 [报告]
发表于 2013-03-01 00:13 |只看该作者
使用以下测试程序,便于理解。

测试一:使用pipe2且传入参数的flags为0(相当于使用pipe)
  1. #include <unistd.h>
  2. #include <signal.h>
  3. #include <string.h>
  4. #include <stdio.h>


  5. int fd[2];

  6. void handle(int sig)
  7. {
  8.         if(sig == SIGUSR1)
  9.                 write(fd[1],"p",1);
  10. }

  11. int main(int argc, char **argv)
  12. {
  13.         char c;
  14.         pid_t pid;
  15.        
  16.         if(pipe2(fd, 0) < 0) {
  17.                 printf("can not creat pipe!\n");
  18.                 return -1;
  19.         }
  20.        
  21.         if((pid = fork()) < 0) {
  22.                 printf("can not fork!\n");
  23.                 return -1;
  24.         } else if(pid > 0) {
  25.                 close(fd[0]);
  26.                 signal(SIGUSR1, handle);
  27.                 for(;;) {
  28.                 }
  29.         } else {
  30.                 close(fd[1]);
  31.                 for(;;) {
  32.                         if(read(fd[0], &c, 1) == 1)
  33.                                 printf("c:%c\n", c);
  34.                         else
  35.                                 printf("nothing to read!\n");

  36.                         sleep(2);
  37.                 }
  38.         }

  39.         return 0;
  40. }
复制代码
编译后,可以看到程序阻塞在那里,当使用“kill -10 进程号”时,才会从管道中读出数据。


测试二,使用pipe2时设置flags的参数为O_NONBLOCK
  1. #include <sys/types.h>
  2. #include <sys/stat.h>
  3. #include <fcntl.h>
  4. #include <unistd.h>
  5. #include <signal.h>
  6. #include <string.h>
  7. #include <stdio.h>


  8. int fd[2];

  9. void handle(int sig)
  10. {
  11.         if(sig == SIGUSR1)
  12.                 write(fd[1],"p",1);
  13. }

  14. int main(int argc, char **argv)
  15. {
  16.         char c;
  17.         pid_t pid;
  18.        
  19.         if(pipe2(fd, O_NONBLOCK) < 0) {
  20.                 printf("can not creat pipe!\n");
  21.                 return -1;
  22.         }
  23.        
  24.         if((pid = fork()) < 0) {
  25.                 printf("can not fork!\n");
  26.                 return -1;
  27.         } else if(pid > 0) {
  28.                 close(fd[0]);
  29.                 signal(SIGUSR1, handle);
  30.                 for(;;) {
  31.                 }
  32.         } else {
  33.                 close(fd[1]);
  34.                 for(;;) {
  35.                         if(read(fd[0], &c, 1) == 1)
  36.                                 printf("c:%c\n", c);
  37.                         else
  38.                                 printf("nothing to read!\n");

  39.                         sleep(2);
  40.                 }
  41.         }

  42.         return 0;
  43. }
复制代码
编译后,执行程序可以看到它不停的打印消息,若管道内没有数据的话,read就直接返回了。

论坛徽章:
208
巨蟹座
日期:2013-09-02 09:16:36卯兔
日期:2013-09-02 20:53:59酉鸡
日期:2013-09-05 21:21:45戌狗
日期:2013-10-15 20:51:17寅虎
日期:2013-10-18 21:13:16白羊座
日期:2013-10-23 21:15:19午马
日期:2013-10-25 21:22:48技术图书徽章
日期:2013-11-01 09:11:32双鱼座
日期:2013-11-01 20:29:44丑牛
日期:2013-11-01 20:40:00卯兔
日期:2013-11-11 09:21:32酉鸡
日期:2013-12-04 19:56:39
5 [报告]
发表于 2013-03-01 09:11 |只看该作者
txgc_wm 发表于 2013-02-28 23:39
read依赖于管道的打开模式,打开管道时可使用pipe2设定相应的flags。书上所写的阻塞是在管道中没有数据的情 ...

windows又如何处理?

论坛徽章:
0
6 [报告]
发表于 2013-03-01 09:31 |只看该作者
回复 5# 流氓无产者

不知楼主问的东西指的是什么? windows上的东西我不懂诶。

论坛徽章:
4
水瓶座
日期:2013-09-06 12:27:30摩羯座
日期:2013-09-28 14:07:46处女座
日期:2013-10-24 14:25:01酉鸡
日期:2014-04-07 11:54:15
7 [报告]
发表于 2013-03-01 13:19 |只看该作者
管道做进程间通信, 两个管道, 一个父到子, 一个子到父.

子阻塞读, 父写1字节唤醒之, 然后父读另一个管道, 然后子写1字节唤醒之.

论坛徽章:
208
巨蟹座
日期:2013-09-02 09:16:36卯兔
日期:2013-09-02 20:53:59酉鸡
日期:2013-09-05 21:21:45戌狗
日期:2013-10-15 20:51:17寅虎
日期:2013-10-18 21:13:16白羊座
日期:2013-10-23 21:15:19午马
日期:2013-10-25 21:22:48技术图书徽章
日期:2013-11-01 09:11:32双鱼座
日期:2013-11-01 20:29:44丑牛
日期:2013-11-01 20:40:00卯兔
日期:2013-11-11 09:21:32酉鸡
日期:2013-12-04 19:56:39
8 [报告]
发表于 2013-03-01 19:07 |只看该作者
txgc_wm 发表于 2013-03-01 09:31
回复 5# 流氓无产者

不知楼主问的东西指的是什么? windows上的东西我不懂诶。

windows上用pipe处理命令,如何做到异步

论坛徽章:
1
白羊座
日期:2013-10-17 23:11:46
9 [报告]
发表于 2013-03-03 21:59 |只看该作者
回复 3# txgc_wm


    有数据的吧,父子进程都往管道里写字符了。

论坛徽章:
0
10 [报告]
发表于 2013-03-04 11:24 |只看该作者
回复 9# popkart718

你可以用测试例子试试,再看看我说的那个内核文件。


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

本版积分规则 发表回复

  

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

清除 Cookies - ChinaUnix - Archiver - WAP - TOP