Когда вы выполните программу, то должны получить вывод, похожий на следующий (полученный на одной из машин авторов):
$ ./popen1
Output was:-
Linux suse103 2.6.20.2-2-default #1 SMP Fri Mar 9 21:54:10 UTC 2001 i686 i686 i386 GNU/Linux
Как это работает
Программа применяет функцию
popen
для вызова команды
uname
с параметром
– а
. Затем она использует возвращенный файловый поток для чтения данных, до
BUFSIZ
символов (как задано в директиве
#define
из файла stdio.h), и затем выводит их на экран. Поскольку вы перехватываете вывод команды uname внутри программы, его можно обрабатывать.
Отправка вывода в popen
Теперь, когда вы рассмотрели пример захвата вывода из внешней программы, давайте познакомимся с отправкой вывода во внешнюю программу. В упражнении 13.2 показана программа popen2.c, передающая по каналу данные другой программе. В этом примере будет использована команда od (от англ. octal dump — восьмеричный дамп).
Упражнение 13.2. Пересылка вывода в другую программу
Взглянув на следующий программный код, вы увидите, что он очень похож на предыдущий пример, за исключением того, что вы пишете данные в канал вместо чтения данных из него. Далее приведена программа popen2.c.
#include <unistd.h>
#include <stdlib.h>
#include <stdio.h>
#include <string.h>
int main {
FILE *write_fp;
char buffer[BUFSIZ + 1];
sprintf(buffer, "Once upon a time, there was...\n");
После выполнения этой программы вы должны получить следующий вывод:
$ ./popen2
0000000 O n c e u p o n a t i m e
0000020 , t h e r e w a s . . . \n
0000037
Как это работает
Программа применяет
popen
с параметром "w" для запуска команды
od -с
таким образом, что может отправить данные этой команде. Затем она отправляет строку, которую команда
od -с
получает и обрабатывает; далее команда
od -с
выводит результат обработки в своем стандартном выводе.
Такой же вывод можно получить из командной строки с помощью следующей команды:
$ echo "Once upon a time, there was..." | od -c
Передача данных большого объема
Механизм, применявшийся до сих пор, просто отправляет и получает все данные в одном вызове
fread
или
fwrite
. Порой вам может понадобиться отправлять данные меньшими порциями или вы не будете знать размера вывода. Для того чтобы не объявлять слишком большой буфер, можно просто применить множественные вызовы