3

我有一个程序,我从几个子进程写入管道,然后尝试从每个管道读取写入每个进程的所有消息,并将它们打印到屏幕上。使用以下代码(特别是使用read系统调用将消息存储到缓冲区的 while 循环buf),我的程序将挂起而不退出,也不会打印发送到不同进程的所有消息。

for (i = 0; i < MAXP; i++) {
    if(id == i) {
        while(read(pfds[i][0], buf, sizeof(buf)) > 0)
             printf("process%d has received a message from %s\n",i,buf);    
    }
}

但是,使用以下代码,程序会正确退出,但不会打印所有消息(因为它们并未全部读取):

for (i = 0; i < MAXP; i++) {
    if(id == i) {
        nbytes = read(pfds[i][0], buf, sizeof(buf));
        printf("process%d has received a message from %s\n",i,buf); 
    }
}

这是我写入管道的代码:

write(pfds[j][1],msg,9);  // write the message to j pipe

消息是:

sprintf(msg, "process%d", i); // create the message - 9 bytes (inc. null term)
// the message is "process0" or "process1" ... through "process8"

这是一个 9 字节的 char 数组:

char msg[9];

解决方案

for (j = 0; j < MAXP; j++) {
    close(pfds[j][1]); // close write to j from i
}
4

2 回答 2

5

也许b / c你没有关闭管道?如果写端没有关闭,读将继续挂起

尝试添加

close(pfds[j][1]);

在你把你的味精写到管道之后

于 2013-11-19T21:32:41.453 回答
2

read(fd)(管道的读取端在哪里)只有在管道的所有写入端都已关闭时fd才会返回。0

请注意,fork()-ing 将复制所有打开的文件描述符。所以这:

int fd[2];
pipe(fd);
if (fork() == 0) {
  write(fd[1], "abc", 3);
  close(fd[1]);
} else {
  char buf[100];
  while (read(fd[0], buf, 100) > 0) {
    // ...
  }
}

将导致无限循环(因为fd[1]在父进程中仍处于打开状态),而这:

int fd[2];
pipe(fd);
if (fork() == 0) {
  write(fd[1], "abc", 3);
  close(fd[1]);
} else {
  close(fd[1]);

  char buf[100];
  while (read(fd[0], buf, 100) > 0) {
    // ...
  }
}

应该终止。

于 2013-11-19T21:38:32.007 回答