popen 实用实战

popen和system都可以执行外部命令。
popen相当于是先创建一个管道,fork,关闭管道的一端,执行exec,返回一个标准的io文件指针。
system相当于是先后调用了fork, exec,waitpid来执行外部命令
popen本身是不阻塞的,要通过标准io的读取使它阻塞
system本身就是阻塞的。
最近写的程序,要求进程在调用的外部命令运行完毕之后,再继续 向下进行。
一开始调用的popen,然后只是用了fgetc,使其阻塞,但是总是阻塞不了。原因就是如果外部命令有很多的输出内容,那fgets在得到输出的第一个字符的时候就返回了,不在阻塞了;调用fread,如果size和nitems设置的不够大,也是一样的问题。比如外部命令要输出100个字符,结果size是sizeof(char),nitems是10,那么当fread读到地10个字符的时候,就已经满足条件了,就返回了。
正确的方法是调用system,因为system最后会调用waitpid,来等待子进程运行完毕。
其实可以调用pclose来阻塞
 
下面看一下代码示例:
test_log:
#include <stdio.h>
#include<string.h>
#include <unistd.h>

int main(void)
{
    int t = 20;
    while(t-- > 0) {
        sleep(1);
        fprintf(stdout, "run test log %d\n", t);
        fflush(stdout); //这里不flush的话,popen可能无法实时读到这里输出的内容
    }
    return 0;
}
popen使用示例:
int main(void)
{
    char buf[LOG_MAX_PER_LEN];
    FILE* fp = popen("test_log", "r");
    if(fp) {
        printf("open success\n");
        while(fgets(buf, LOG_MAX_PER_LEN, fp) != NULL) {
            if(buf[strlen(buf) - 1] == '\n') {
                buf[strlen(buf) - 1] = '\0';
            }
            fprintf(stdout, "[test log output]: %s\n", buf);
        }
    } else {
        perror("popen fail!\n");
        return -1;
    }

    int ret = pclose(fp);
    printf("pclose ret %d\n", ret);
    return 0;
}

 

posted on 2022-08-27 16:56  疾速瓜牛  阅读(1171)  评论(0编辑  收藏  举报

导航