パイプを使用して2つのプログラム間で単純な文字列を送信する方法は?


111

ネットで検索してみましたが、リソースがほとんどありません。小さな例で十分です。

編集つまり、互いに通信する2つの異なるCプログラムです。1つのプログラムは "Hi"を送信し、他のプログラムはそれを受信する必要があります。そんな感じ。

c  unix  pipe 

1
おそらくあなたは何かのような意味ではありませんls | grep ".o"か?おそらく、あなたが何を意味するのかについてもう少し説明すると役立つでしょう...
Jerry Coffin

13
人に来て...少し努力。グーグルの「c pipes example code」。最初の結果は正確です:tldp.org/LDP/lpg/node11.html
Stephen

4
2つの完全に異なるプログラム間の通信が必要です。そのためのリソースを見つけることができませんでした。

1
プロセスをフォークしていない場合は、「名前付きパイプ」を調べる必要があります。
Maygarden裁判官、10

回答:


156

通常のパイプは、2つの関連プロセスのみを接続できます。これはプロセスによって作成され、最後のプロセスがそれを閉じると消えます。

名前付きパイプ、また、その動作のためのFIFOと呼ばは、2つの無関係なプロセスを接続するために使用され、独立のプロセスが存在することができます。つまり、誰も使用していない場合でも存在できるということです。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
おそらく、1つ以上の親子関係を介して関連付けられているプロセス(兄弟を含む)。共通の祖先がパイプの両端を作成したはずです。無関係なプロセスには、その共通の祖先がありません。
MSalters 2014年

4
リーダーが最初にキックオフすると、これは機能しません。簡単な修正はopen()、リーダーのをループ内に置くことです。ただし、2つのプログラムの例を提供するため、+ 1。
gsamaras 2014年

この例では、Windowsで作業するために微調整が必​​要ですか?unistd.hはPOSIXであり、すべて...
David Karlsson

はい、Windows用に調整する必要があります。名前付きパイプ上のWikipediaの記事は、 UNIX / Windowsの違いとクイックのいくつかについて説明し、Googleの検索 Windowsの実装と缶の助けを。
jschmier 2015年

41

これは、Cでのパイプの作成から、プログラムをフォークしてパイプを使用する方法を示しています。fork()したくない場合は、名前付きパイプを使用できます。

さらに、のprog1 | prog2出力prog1をstdoutに送信しstdin、in から読み取ることで、効果を得ることができますprog2。という名前のファイルを開いて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
とにかく、このコードを2つの異なる関数に使用できますか?パイプへの書き込みが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()子プロセスの標準入力への書き込みに通常のファイルとしてパイプを使用するか、またはコールは、その標準出力から読み取ります。
  3. pclose()パイプを閉じ、子プロセスが終了しますコール。

「コンバーター」プログラムが別のプログラムであることは認めますが、この例は問題の要点を逃していると思います。最初のコメントは、兄弟/親/従兄弟関係を持たない完全に独立したプログラム間の通信を扱います。
cmm 2015年

2

まず、プログラム1に文字列を書き込んでもらいますstdout(画面に表示したい場合と同じように)。次に、2番目のプログラムはstdin、ユーザーがキーボードから入力しているかのように、から文字列を読み取る必要があります。次に実行します:

$ program_1 | program_2

1

この回答は、将来のGoogle社員にとって役立つかもしれません。

#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;

}

ここでは、高度な双方向パイプ呼び出しの例を見つけることができます。

弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.