在c中重定向孩子的stdin和stdout

时间:2015-01-02 00:27:31

标签: c linux pipe

我需要创建一个客户端 - 服务器应用程序(使用套接字),基本思路是:

  • 客户端发送字符串
  • 服务器接收字符串并将其作为另一个应用程序的标准输入
  • 发送
  • 服务器读取应用程序的标准输出
  • 服务器向客户发送答案。

“其他应用”是一个封闭的源计算(一个计算器从标准输入读取4 5 +并打印9)。

我正在尝试在服务器上创建一个双管道,fork,并使用这个管道来重定向计算的stdin和stdout:

if(!fork())
{

    close(STDOUT_FILENO);
    close(STDIN_FILENO);

    dup2(outfd[0], STDIN_FILENO);
    dup2(infd[1], STDOUT_FILENO);

    close(outfd[0]); /* Not required for the child */
    close(outfd[1]);
    close(infd[0]);
    close(infd[1]);

    system("./calc/calc ");
    exit(0);
}

并从服务器写入一个管道并读取另一个

    close(outfd[0]); /* These are being used by the child */
    close(infd[1]);

    char c;
    do
    {
        c=getchar();
        write(outfd[1],&c,1);
    }while(c!='\n');
    input[read(infd[0],input,100)] = 0; /* Read from child’s stdout */

    printf("%s",input);

    close(outfd[1]);
    close(infd[0]);

我知道这是非常不完整的,但我认为至少应该在calc中输出第一行的输出。

(完整代码)

#include <unistd.h>
#include <stdio.h>
#include <stdlib.h>

int main()
{

int outfd[2];
int infd[2];

pipe(outfd); /* Where the parent is going to write to */
pipe(infd); /* From where parent is going to read */

if(!fork())
{

    close(STDOUT_FILENO);
    close(STDIN_FILENO);

    dup2(outfd[0], STDIN_FILENO);
    dup2(infd[1], STDOUT_FILENO);

    close(outfd[0]); /* Not required for the child */
    close(outfd[1]);
    close(infd[0]);
    close(infd[1]);

    //system("/usr/bin/bc -q");
    system("./calc/calc ");
    exit(0);
}
else
{

    char input[100];

    close(outfd[0]); /* These are being used by the child */
    close(infd[1]);

    //write(outfd[1],"2^32\n",5); /* Write to child’s stdin */
    //write(outfd[1],"2 4 +\n",6);
    char c;
    do
    {
        c=getchar();
        write(outfd[1],&c,1);
    }while(c!='\n');
    input[read(infd[0],input,100)] = 0; /* Read from child’s stdout */

    printf("%s",input);

    close(outfd[1]);
    close(infd[0]);

}
return 0;

这是非常基本的,但它只是用于测试。当我执行它时,它什么都不做。

如果我ps -e | grep calc在执行期间,我可以看到它正在运行,但无论我输入什么似乎都没有做任何事情。

1 个答案:

答案 0 :(得分:1)

尝试移动:

close(outfd[1]);

do-while循环和input[read(infd[0],input,100)] = 0;

之间

这将在EOF的标准输入上发送calc。它将退出,这将刷新其输出缓冲区,因此将有一些东西需要阅读。

您的代码已死锁:您在关闭管道之前等待输出,但它在发送输出之前等待您关闭管道。