转载自:https://blog.youkuaiyun.com/stone8761/article/details/77498439
popen() 可以执行 shell 命令,并读取此命令的返回值
popen() 函数通过创建一个管道,调用 fork() 产生一个子进程,调用一个 shell 以运行命令来开启一个进程。可以通过这个管道执行标准输入输出操作。这个管道必须由 pclose() 函数关闭,必须由 pclose() 函数关闭,必须由 pclose() 函数关闭,而不是 fclose() 函数(若使用 fclose() 则会产生僵尸进程)。pclose() 函数关闭标准 I/O 流,等待命令执行结束,然后返回 shell 的终止状态。如果 shell 不能被执行,则 pclose() 返回的终止状态与 shell 已执行 exit 一样。
函数原型
FILE *popen(const char *command, const char *type);
int pclose(FILE *stream);
type 参数只能是读或者写中的一种,得到的返回值(标准 I/O 流)也具有和 type 相应的只读或只写类型。如果 type 是 “r” 则文件指针连接到 command 的标准输出;如果 type 是 “w” 则文件指针连接到 command 的标准输入。
command 参数是一个指向以 NULL 结束的 shell 命令字符串的指针。这行命令将被传到 bin/sh 并使用 -c 标志,shell 将执行这个命令。
popen() 的返回值是个标准 I/O 流,必须由 pclose() 来终止。前面提到这个流是单向的(只能用于读或写)。向这个流写内容相当于写入该命令的标准输入,命令的标准输出和调用 popen() 的进程相同;与之相反的,从流中读数据相当于读取命令的标准输出,命令的标准输入和调用 popen() 的进程相同。
返回值
如果调用 fork()(产生子进程) 或 pipe()(申请管道) 失败,或者不能分配内存将返回 NULL,否则返回标准 I/O 流。popen() 没有为内存分配失败设置 errno 值。如果调用 fork() 或 pipe() 时出现错误,errno 被设为相应的错误类型。如果 type 参数不合法,errno 将返回 EINVAL。
例:
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
void print_result(FILE *fp)
{
char buf[100];
if(!fp) {
return;
}
printf("\n>>>\n");
while(memset(buf, 0, sizeof(buf)), fgets(buf, sizeof(buf) - 1, fp) != 0 ) {
printf("%s", buf);
}
printf("\n<<<\n");
}
int main(void)
{
FILE *fp = NULL;
while(1) {
fp = NULL;
fp = popen("ls", "r");
if(!fp) {
perror("popen");
exit(EXIT_FAILURE);
}
print_result(fp);
pclose(fp);
sleep(1);
}
}