Linux环境下用popen()函数读取shell命令控制台输出

Unix界有一句名言:“一行shell脚本胜过万行C程序”,虽然这句话有些夸张,但不可否认的是,借助脚本确实能够极大的简化一些编程工作。Linux环境下,利用popen()函数可以执行shell命令输出结果。

一、 popen()函数定义

#include <stdio.h>
FILE *popen(const char *command, const char *type);
int pclose(FILE *stream);

二、 popen()函数说明
popen() 函数通过创建一个管道,调用 fork 产生一个子进程,执行一个 shell 以运行命令来开启一个进程。这个进程必须由 pclose() 函数关闭,而不是 fclose() 函数。pclose() 函数关闭标准 I/O 流,等待命令执行结束,然后返回 shell 的终止状态。如果 shell 不能被执行,则 pclose() 返回的终止状态与 shell 已执行 exit 一样。

type 参数只能是读或者写中的一种,得到的返回值(标准 I/O 流)也具有和 type 相应的只读或只写类型。如果 type 是 “r” 则文件指针连接到 command 的标准输出;如果 type 是 “w” 则文件指针连接到 command 的标准输入。

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

popen 的返回值是个标准 I/O 流,必须由 pclose 来终止。前面提到这个流是单向的。所以向这个流写内容相当于写入该命令的标准输入;命令的标准输出和调用 popen 的进程相同。与之相反的,从流中读数据相当于读取命令的标准输出;命令的标准输入和调用 popen 的进程相同。

三、 popen()函数返回值
如果调用 fork() 或 pipe() 失败,或者不能分配内存将返回NULL,否则返回标准 I/O 流。

四、 popen()函数返回错
popen 没有为内存分配失败设置 errno 值。
如果调用 fork() 或 pipe() 时出现错误,errno 被设为相应的错误类型。
如果 type 参数不合法,errno将返回EINVAL。

五、 popen()函数示例

#include <stdio.h>
#include <unistd.h>

#define BUFFER_MAX_LEN 4096

int main(void)
{
	char buffer[BUFFER_MAX_LEN];

	FILE *fp = popen("ls -il ~", "r");
	if(fp == NULL) 
	{
		printf("popen error\n");
		return 1;
	}

	while(fgets(buffer, BUFFER_MAX_LEN, fp) != NULL)
		printf("%s", buffer);

	pclose(fp);

	return 0;
}

六、 运行结果

[ycxie@fedora Workspace]$ gcc getshell.c -o getshell -Wall
[ycxie@fedora Workspace]$ ./getshell
total 36
1446231 drwxr-xr-x. 2 ycxie ycxie 4096 Mar  2 23:50 Desktop
1446235 drwxr-xr-x. 2 ycxie ycxie 4096 Mar  2 23:50 Documents
1446232 drwxr-xr-x. 2 ycxie ycxie 4096 Mar  2 23:50 Downloads
1446236 drwxr-xr-x. 2 ycxie ycxie 4096 Mar  2 23:50 Music
1446237 drwxr-xr-x. 2 ycxie ycxie 4096 Mar  2 23:50 Pictures
1446234 drwxr-xr-x. 2 ycxie ycxie 4096 Mar  2 23:50 Public
1446233 drwxr-xr-x. 2 ycxie ycxie 4096 Mar  2 23:50 Templates
1446238 drwxr-xr-x. 2 ycxie ycxie 4096 Mar  2 23:50 Videos
1446309 drwxr-xr-x. 2 ycxie ycxie 4096 Mar 21 11:09 Workspace

Leave a Reply

Your email address will not be published. Required fields are marked *