在做项目时,需要在进程中执行一个shell命令,最先想到的便是fork()一个子进程再去execl执行命令,后来发现了popen函数,简单化了这个过程。当然,popen创建的也是子进程。popen函数会创建一个管道用于父子进程间通信。父进程通过管道读信息或者写信息,至于是读还是写取决于父进程调用popen时传递的参数(r、w)
一、函数定义
头文件:stdio.h
函数:FILE *popen(const char *command, const char *type)
函数功能:popen()会调用fork()产生子进程,然后从子进程中调用/bin/sh -c来执行参数command的指令。参数type可使用“r”代表读取,“w”代表写入。如果type是"r"则文件指针连接到command的标准输出,读取命令的标准输出;如果type是"w"则文件指针连接到command的标准输入,写入命令的标准输入。依照此type值,popen()会建立管道连到子进程的标准输出设备或标准输入设备,然后返回一个文件指针。随后进程便可利用此文件指针来读取子进程的标准输出设备或是写入到子进程的标准输入设备中
返回值:若成功则返回标准I/O流文件指针(FILE *),否则返回NULL,错误原因存于errno中
函数:int pclose(FILE * stream);
函数功能:函数功能:pclose()用来关闭由popen所建立的管道及文件指针。参数stream为先前由popen()所返回的文件指针
返回值:若成功返回shell的终止状态(即子进程的终止状态),若出错返回-1,错误原因存于errno中
二、简单应用
在程序中想执行以下ping测试:
ping -I eth0 -c 2 www.baidu.com
ping_test.c
#include <stdio.h>
#include <errno.h>
#include <string.h>
int main(int argc,char **argv)
{
int rc = 0; // 用于接收命令返回值
int rv=-1;
FILE *fp;
char command[128];
char result_buf[1024];
/*将要执行的命令写入buf*/
snprintf(command, sizeof(command), "ping -I eth0 -c 2 www.baidu.com");
printf("command=%s\n",command);
/*执行预先设定的命令,并读出该命令的标准输出*/
fp = popen(command, "r"); //这时执行shell命令的相关内容不会出现在标准输出上
if( NULL == fp )
{
printf("popen ping_test failure.\n");
return -1;
}
printf("popen ping_test successfully!\n");
/*读ping的内容,因为fp是标准I/O流文件指针,所以要用fread来读*/
if( (rv=fread(result_buf,1,sizeof(result_buf),fp))<0 )
{
printf("Read from fp failure:%s\n",strerror(errno));
return -2;
}
printf("result_buf=%s\n",result_buf);
rc = pclose(fp); //关闭文件指针
if(-1 == rc)
{
printf("close file point failure.\n");
return -3;
}
return 0;
}
结果: