Як надіслати просту рядок між двома програмами за допомогою труб?


111

Я спробував шукати в мережі, але майже немає ресурсів. Невеликого прикладу було б достатньо.

EDIT Я маю на увазі дві різні програми C, що спілкуються між собою. Одна програма повинна надіслати "Привіт", а інша повинна її отримати. Щось схоже.

c  unix  pipe 

1
Імовірно, ви не маєте на увазі щось подібне ls | grep ".o"? Можливо, трохи більше пояснень того, що ви маєте на увазі, допоможе ...
Джеррі Коффін

13
Давай людино ... трохи зусиль. Google "код коду прикладу". Перший результат точний: tldp.org/LDP/lpg/node11.html
Стівен

4
Я хочу спілкування між двома абсолютно різними програмами. Я не зміг знайти для цього ресурс.

1
Якщо ви не розгалужуєте процес, то вам потрібно подивитися на "названі труби".
Суддя Мейґарден

Відповіді:


156

Звичайна труба може з'єднувати лише два пов'язані з цим процеси. Він створений процесом і зникне, коли останній процес його закриє.

Іменований канал , званий також буфер FIFO для його поведінки, може бути використаний для з'єднання двох незв'язаних процесів і існує незалежно від процесів; тобто він може існувати, навіть якщо його ніхто не використовує. FIFO створюється за допомогою функції mkfifo()бібліотеки.

Приклад

письменник.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;
}

читач.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
Що вважають суміжними процесами ?
Пітікос

7
Ймовірно, процеси, пов’язані через одного або декількох стосунків батьків / дитини (наприклад, включають братів і сестер). Спільний предок створив би два кінці труби. Неспорідненим процесам бракує цього спільного предка.
MSalters

4
Це не спрацює, якщо читач стартує першим. Швидке виправлення полягатиме в тому, щоб вставити open()читач всередину циклу. Однак +1, оскільки ви надаєте приклад двох програм.
gsamaras

Я вважаю, що цей приклад потребує певного налаштування, щоб працювати на windows? unistd.h будучи POSIX і все ...
David Karlsson

Так, для Windows знадобиться налаштування. У статті Вікіпедії про названі труби обговорюються деякі відмінності Unix / Windows, і швидкий пошук Google може допомогти у впровадженні Windows.
jschmier

41

З Створення Pipes в C , це показує вам , як розщедритися програму , щоб використовувати трубу. Якщо ви не хочете роздвоювати (), ви можете використовувати названі труби .

Крім того, ви можете отримати ефект prog1 | prog2, посилаючи висновок prog1на стандартний висновок і читання з stdinв 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
Ей, Стівен, все-таки я можу використовувати цей код для двох різних функцій? сенс запис у трубу виконується в одній функції, а читання труби в іншій функції ?? такий робочий код буде вдячний.
Мохсін

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()Виклик , який закриває трубу і викликає дочірній процес завершується.

Я думаю, що цей приклад пропускає суть питання, хоча я даю, що програма "конвертер" - це інша програма. Перший коментар стосується спілкування між абсолютно незалежними програмами, які не мають стосунків брат / сестра / батько / другий родич.
смм

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;

}

Ви можете знайти просунутий двосторонній приклад виклику труби тут .

Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.