如何使用管道在两个程序之间发送简单的字符串?


111

我尝试在网上搜索,但几乎没有任何资源。一个小例子就足够了。

编辑我的意思是,两个不同的C程序彼此通信。一个程序应发送“ Hi”,而另一个程序应接收。这样的事情。

c  unix  pipe 

1
想必你不是在说什么ls | grep ".o"吗?也许更多地解释您的意思会有所帮助...
杰里·科芬

13
来吧...一点点努力。谷歌“ c管道示例代码”。第一个结果是确切的:tldp.org/LDP/lpg/node11.html
斯蒂芬

4
我想要两个完全不同的程序之间的通信。我无法为此找到资源。

1
如果您不打算分叉进程,则需要查看“命名管道”。
Maygarden法官10年

Answers:


156

常规管道只能连接两个相关过程。它由一个进程创建,并在最后一个进程关闭时消失。

命名管道,也称为FIFO为其行为,可以用于连接两个不相关的进程和独立的处理的存在; 意味着即使没有人使用它,它也可以存在。使用mkfifo()库函数创建一个FIFO 。

writer.c

#include <fcntl.h>
#include <sys/stat.h>
#include <sys/types.h>
#include <unistd.h>

int main()
{
    int fd;
    char * myfifo = "/tmp/myfifo";

    /* create the FIFO (named pipe) */
    mkfifo(myfifo, 0666);

    /* write "Hi" to the FIFO */
    fd = open(myfifo, O_WRONLY);
    write(fd, "Hi", sizeof("Hi"));
    close(fd);

    /* remove the FIFO */
    unlink(myfifo);

    return 0;
}

reader.c

#include <fcntl.h>
#include <stdio.h>
#include <sys/stat.h>
#include <unistd.h>

#define MAX_BUF 1024

int main()
{
    int fd;
    char * myfifo = "/tmp/myfifo";
    char buf[MAX_BUF];

    /* open, read, and display the message from the FIFO */
    fd = open(myfifo, O_RDONLY);
    read(fd, buf, MAX_BUF);
    printf("Received: %s\n", buf);
    close(fd);

    return 0;
}

注意:为简单起见,上述代码中省略了错误检查。


6
什么是相关过程
Pithikos 2014年

7
可能是通过一个或多个父/子关系(例如,包括兄弟姐妹)关联的过程。共同祖先将创建管道的两端。不相关的流程缺少该共同祖先。
MSalters 2014年

4
如果读者先开始,这将不起作用。一种快速的解决方案是将open()阅读器的放入循环中。但是+1是因为您提供了两个程序示例。
gsamaras

我认为这个示例需要一些调整才能在Windows上工作?unistd.h是POSIX和所有...
David Karlsson,2015年

是的,它需要针对Windows进行调整。该命名管道维基百科的文章讨论了一些UNIX / Windows的差异和快速谷歌搜索与Windows实现可以提供帮助。
jschmier

41

用C创建管道,这展示了如何fork一个程序使用的管道。如果您不想fork(),则可以使用命名管道

另外,您可以prog1 | prog2通过将输出发送prog1到stdout并从stdinin中读取来获得效果prog2。您也可以通过打开一个名为stdin的文件来读取stdin /dev/stdin(但不确定该文件的可移植性)。

/*****************************************************************************
 Excerpt from "Linux Programmer's Guide - Chapter 6"
 (C)opyright 1994-1995, Scott Burkett
 ***************************************************************************** 
 MODULE: pipe.c
 *****************************************************************************/

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <sys/types.h>

int main(void)
{
        int     fd[2], nbytes;
        pid_t   childpid;
        char    string[] = "Hello, world!\n";
        char    readbuffer[80];

        pipe(fd);

        if((childpid = fork()) == -1)
        {
                perror("fork");
                exit(1);
        }

        if(childpid == 0)
        {
                /* Child process closes up input side of pipe */
                close(fd[0]);

                /* Send "string" through the output side of pipe */
                write(fd[1], string, (strlen(string)+1));
                exit(0);
        }
        else
        {
                /* Parent process closes up output side of pipe */
                close(fd[1]);

                /* Read in a string from the pipe */
                nbytes = read(fd[0], readbuffer, sizeof(readbuffer));
                printf("Received string: %s", readbuffer);
        }

        return(0);
}

1
嗨,斯蒂芬,无论如何,我可以将此代码用于两个不同的功能?意思是写入管道是在一个功能中完成,而读取管道是在另一个功能中?这样的工作代码将不胜感激。
Mohsin

8
dup2( STDIN_FILENO, newfd )

并阅读:

char reading[ 1025 ];
int fdin = 0, r_control;
if( dup2( STDIN_FILENO, fdin ) < 0 ){
    perror( "dup2(  )" );
    exit( errno );
}
memset( reading, '\0', 1025 );
while( ( r_control = read( fdin, reading, 1024 ) ) > 0 ){
    printf( "<%s>", reading );
    memset( reading, '\0', 1025 );
}
if( r_control < 0 )
    perror( "read(  )" );    
close( fdin );    

但是,我认为这fcntl可能是更好的解决方案

echo "salut" | code

6

一个程序写入stdout的内容可以被另一个通过stdin读取。简而言之,使用c来编写,使用prog1来打印内容,printf()并使用prog2来读取内容scanf()。然后就跑

./prog1 | ./prog2

4

这是一个示例

int main()
{
    char buff[1024] = {0};
    FILE* cvt;
    int status;
    /* Launch converter and open a pipe through which the parent will write to it */
    cvt = popen("converter", "w");
    if (!cvt)
    {
        printf("couldn't open a pipe; quitting\n");
        exit(1)
    }
    printf("enter Fahrenheit degrees: " );
    fgets(buff, sizeof (buff), stdin); /*read user's input */
    /* Send expression to converter for evaluation */
    fprintf(cvt, "%s\n", buff);
    fflush(cvt);
    /* Close pipe to converter and wait for it to exit */
    status=pclose(cvt);
    /* Check the exit status of pclose() */
    if (!WIFEXITED(status))
        printf("error on closing the pipe\n");
    return 0;
}

该程序的重​​要步骤是:

  1. popen()在子进程和父进程中的管道之间建立关联的调用。
  2. fprintf()使用管道作为普通文件调用,以写入子进程的stdin或从其stdout中读取。
  3. pclose()关闭管道并导致子进程终止的调用。

我认为这个示例没有提出问题,尽管我同意“转换器”程序是一个不同的程序。第一条评论涉及没有兄弟姐妹/父母/第二堂兄弟关系的完全独立程序之间的通信。
cmm 2015年

2

首先,让程序1将字符串写入其中stdout(就像您希望它出现在屏幕上一样)。然后,第二个程序应该从中读取一个字符串stdin,就像用户正在从键盘上键入一样。然后运行:

$ program_1 | program_2

1

这个答案可能对将来的Googler有帮助。

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

int main(){     
     int p, f;  
     int rw_setup[2];   
     char message[20];      
     p = pipe(rw_setup);    
     if(p < 0){         
        printf("An error occured. Could not create the pipe.");  
        _exit(1);   
     }      
     f = fork();    
     if(f > 0){
        write(rw_setup[1], "Hi from Parent", 15);    
     }  
     else if(f == 0){       
        read(rw_setup[0],message,15);       
        printf("%s %d\n", message, r_return);   
     }  
     else{      
        printf("Could not create the child process");   
     }      
     return 0;

}

您可以在此处找到高级的双向管道调用示例。

By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.