管道是Linux系统上两个进程之间的一种通信方式。函数popen()复制了一个子进程,并创建一个连接这个子进程的管道.根据管道的方向,子进程可以通过读自己的标准输入来读管道,或者通过写自己的标准输出写管道.
popen()函数调用需要一个命令行字符串,并定义管道的读写属性.它就能像由函数fope()打开的一样,进行使用.任何能够用于读文件的函数,都能够用于读管道.唯一不同的是,必须用函数pclose()而不是fclose()来关闭管道.
下面的例子用于实现:命令ls将运行结果写到标准输出,因此可以在程序中创建管道获取ls的输出.
/* pipein.c */
#include <stdio.h>
int main(int argc, char *argv[])
{
FILE *pd;
char instring[100];
if ((pd = popen("ls \n", "r")) == NULL) {
perror("Cannot open pipe");
exit(1);
}
//注意在函数printf()中不需要换行符,这是由于函数fgets()如同操作文件一样来操作管道,在输入行中已包含了换行符.
while (fgets(instring, sizeof(instring), pd) != NULL) {
printf(" %s", instring);
}
pclose(pd);
return 0;
}
此函数会读取并显示由ls命令找到的文件名.
不带参数运行的cat命令将从标准输入中读取,并输出到标准输出.
/* pipeout.c */
#include <stdio.h>
int main(int argc, char *argv[])
{
FILE *pd;
char instring[100];
if ((pd = popen("cat \n", "w")) == NULL) {
perror("Cannot open pipe");
exit(1);
}
fputs("The first string to cat \n", pd);
fflush(pd);
sleep(5);
fputs("The second string to cat \n", pd);
fflush(pd);
pclose(pd);
return 0;
}
此程序将打开一个管道,传送字符到cat命令,就好像是从终端输入的一样.
由于管道是以”w”方式打开的,程序就可以对它进行写操作,函数fputs()将字符串送到cat命令,后者再将它们写到标准输出,这样字符就显示在屏幕上.调用函数fputs()的原因在于管道是完全缓存的,真正的输出数据只在缓存已满或者关闭管道时才发送.而管道数据的缓存是没有大小限制的,因为它将数据存在磁盘中,所以写管道时永远不会阻塞.这就意味着你可以将所有命令和数据排队,而由子进程按它自身的步调来读取数据.