linux 管道文件 popen,Linux下使用popen()执行shell命令

函数原型:shell

#include “stdio.h”数组

FILE popen( const char command, const char* mode )函数

参数说明:oop

command: 是一个指向以 NULL 结束的 shell 命令字符串的指针。这行命令将被传到 bin/sh 并使用 -c 标志,shell 将执行这个命令。指针

mode: 只能是读或者写中的一种,获得的返回值(标准 I/O 流)也具备和 type 相应的只读或只写类型。若是 type 是 “r” 则文件指针链接到 command 的标准输出;若是 type 是 “w” 则文件指针链接到 command 的标准输入。code

返回值:排序

若是调用成功,则返回一个读或者打开文件的指针,若是失败,返回NULL,具体错误要根据errno判断进程

int pclose (FILE* stream)ip

参数说明:字符串

stream:popen返回的文件指针

返回值:

若是调用失败,返回 -1

做用:

popen() 函数用于建立一个管道:其内部实现为调用 fork 产生一个子进程,执行一个 shell 以运行命令来开启一个进程这个进程必须由 pclose() 函数关闭。

例子:

管道读:先建立一个文件test,而后再test文件内写入“Read pipe successfully !”

#include “stdio.h”

#include “stdlib.h”

int main()

{

FILE *fp;

char buf[200] = {0};

if((fp = popen(“cat test”, “r”)) == NULL) {

perror(“Fail to popen\n”);

exit(1);

}

while(fgets(buf, 200, fp) != NULL) {

printf(“%s”, buf);

}

pclose(fp);

return 0;

}

打印输出: Read pipe successfully !

管道读:

#include “stdio.h”

#include “stdlib.h”

int main()

{

FILE *fp;

char buf[200] = {0};

if((fp = popen(“cat > test1″, “w”)) == NULL) {

perror(“Fail to popen\n”);

exit(1);

}

fwrite(“Read pipe successfully !”, 1, sizeof(“Read pipe successfully !”), fp);

pclose(fp);

return 0;

}

执行完毕后,当前目录下多了一个test1文件,打开,里面内容为Read pipe successfully !

popen()和pclose()

若是你认为上面建立和使用管道的方法过于繁琐的话,你也可使用下面的简单的方法:

库函数:popen()和pclose();

原型:FILEpopen(charcommand,char*type);

返回值:若是成功,返回一个新的文件流。

若是没法建立进程或者管道,返回NULL。

此标准的库函数经过在系统内部调用pipe()来建立一个半双工的管道,而后它建立一个子进程,启动shell,最后在shell上执行command参数中的命令。管道中数据流的方向是由第二个参数type控制的。此参数能够是r或者w,分别表明读或写。但不能同时为读和写。在Linux系统下,管道将会以参数type中第一个字符表明的方式打开。因此,若是你在参数type中写入rw,管道将会以读的方式打开。

虽然此库函数的用法很简单,但也有一些不利的地方。例如它失去了使用系统调用pipe()时能够有的对系统的控制。尽管这样,由于能够直接地使用shell命令,因此shell中的一些通配符和其余的一些扩展符号均可以在command参数中使用。

使用popen()建立的管道必须使用pclose()关闭。其实,popen/pclose和标准文件输入/输出流中的fopen()/fclose()十分类似。

库函数:pclose();

原型:intpclose(FILE*stream);

返回值:返回系统调用wait4()的状态。

若是stream无效,或者系统调用wait4()失败,则返回-1。

注意此库函数等待管道进程运行结束,而后关闭文件流。库函数pclose()在使用popen()建立的进程上执行wait4()函数。当它返回时,它将破坏管道和文件系统。

在下面的例子中,用sort命令打开了一个管道,而后对一个字符数组排序:

#include

#defineMAXSTRS5

intmain(void)

{

intcntr;

FILE*pipe_fp;

char*strings[MAXSTRS]={"echo","bravo","alpha",

"charlie","delta"};

/Createonewaypipelinewithcalltopopen()/

if((pipe_fp=popen("sort","w"))==NULL)

{

perror("popen");

exit(1);

}

/Processingloop/

for(cntr=0;cntr

fputs(strings[cntr],pipe_fp);

fputc('\n',pipe_fp);

}

/Closethepipe/

pclose(pipe_fp);

return(0);

}

由于popen()使用shell执行命令,因此全部的shell扩展符和通配符均可以使用。此外,它还能够和popen()一块儿使用重定向和输出管道函数。再看下面的例子:

popen("ls~scottb","r");

popen("sort>/tmp/foo","w");

popen("sort|uniq|more","w");

下面的程序是另外一个使用popen()的例子,它打开两个管道(一个用于ls命令,另外一个用于

sort命令):

#include

intmain(void)

{

FILE*pipein_fp,*pipeout_fp;

charreadbuf[80];

/Createonewaypipelinewithcalltopopen()/

if((pipein_fp=popen("ls","r"))==NULL)

{

perror("popen");

exit(1);

}

/Createonewaypipelinewithcalltopopen()/

if((pipeout_fp=popen("sort","w"))==NULL)

{

perror("popen");

exit(1);

}

/Processingloop/

while(fgets(readbuf,80,pipein_fp))

fputs(readbuf,pipeout_fp);

/Closethepipes/

pclose(pipein_fp);

pclose(pipeout_fp);

return(0);

}

最后,咱们再看一个使用popen()的例子。此程序用于建立一个命令和文件之间的管道:

#include

intmain(intargc,char*argv[])

{

FILE*pipe_fp,*infile;

charreadbuf[80];

if(argc!=3){

fprintf(stderr,"USAGE:popen3[command][filename]\n");

exit(1);

}

/Open up input file/

if((infile=fopen(argv[2],"rt"))==NULL)

{

perror("fopen");

exit(1);

}

/Create one way pipe line with call topopen()/

if((pipe_fp=popen(argv[1],"w"))==NULL)

{

perror("popen");

exit(1);

}

/Processingloop/

do{

fgets(readbuf,80,infile);

if(feof(infile))break;

fputs(readbuf,pipe_fp);

}while(!feof(infile));

fclose(infile);

pclose(pipe_fp);

return(0);

}

下面是使用此程序的例子:

popen3sortpopen3.c

popen3catpopen3.c

popen3morepopen3.c

popen3catpopen3.c|grepmain

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值