与linux的文件操作中有基于标准I/O操作一样,管道操作也支持基于文件流的操作,这种基于文件流的管道主要用来创建一个连接到另一个进程的管道,这里“另一个进程”是可以执行一定操作的可执行文件,例如用户执行“ls -l”或者./pipe,由于这类操作很常见,所以将一系列创建过程合并到一个函数popen()中完成,这个函数会完成以下步骤:
- 创建一个管道
- fork()一个子进程
- 在父子进程中关闭不需要的文件描述符
- 执行exec()函数族调用
- 执行函数中指定的命令
这个函数可以大大减少代码的编写量,但使用不太灵活,不能自己创建管道那么灵活,并且popen()必须使用标准的I/o函数进行操作,也不能使用read(),wirte()这种不带缓冲的I/O函数,必须使用pclose()来关闭管道流,该函数关闭标准I/O流,并等待命令执行结束
popen()函数语法如下:
pclose()函数语法如下:
这里有个例子使用popen()来执行"ps -ef"命令,popen()函数可以使程序短小精悍
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <fcntl.h>
#define BUFFSIZE 1024
int main()
{
FILE * fp ;
char *cmd = "ps -ef" ;
char buf[BUFFSIZE] ;
//r:产生标准输出,把cmd中的命令送到/bin/sh以-c参数由shell执行
if ((fp = popen(cmd, "r")) == NULL)
{
printf("popen error\n") ;
exit(1) ;
}
while((fgets(buf, BUFFSIZE, fp)) != NULL)//从文件结构指针fp中读取数据,每次读取BUFFSIZE-1个数据
{
printf("%s", buf) ;
}
pclose(fp) ;
exit(0) ;
}
执行效果就是ps -ef的执行效果: