我有一个实用程序,可以输出游戏所需的文件列表。如何在 C 程序中运行该实用程序并获取其输出,以便我可以在同一程序中对其进行操作?

更新:关于缺乏信息的好呼吁。该实用程序会输出一系列字符串,这应该可以在 Mac/Windows/Linux 之间完全移植。请注意,我正在寻找一种编程方式来执行该实用程序并保留其输出(输出到标准输出)。

有帮助吗?

解决方案

对于 Unix 环境中的简单问题,请尝试 popen().

从手册页:

popen() 函数通过创建管道、分叉和​​调用 shell 来打开进程。

如果您使用阅读模式,这正是您所要求的。不知道Windows下是否实现了。

对于更复杂的问题,您需要查找进程间通信。

其他提示

正如其他人指出的那样, popen() 是最标准的方式。由于没有答案提供使用此方法的示例,因此如下:

#include <stdio.h>

#define BUFSIZE 128

int parse_output(void) {
    char *cmd = "ls -l";    

    char buf[BUFSIZE];
    FILE *fp;

    if ((fp = popen(cmd, "r")) == NULL) {
        printf("Error opening pipe!\n");
        return -1;
    }

    while (fgets(buf, BUFSIZE, fp) != NULL) {
        // Do whatever you want here...
        printf("OUTPUT: %s", buf);
    }

    if(pclose(fp))  {
        printf("Command not found or exited with error status\n");
        return -1;
    }

    return 0;
}

Windows 支持 popen,请参见此处:

http://msdn.microsoft.com/en-us/library/96ayss4b.aspx

如果你希望它是跨平台的,popen 是最好的选择。

好吧,假设您在 Windows 环境中使用命令行,则可以使用管道或命令行重定向。例如,

commandThatOutputs.exe > someFileToStoreResults.txt

或者

commandThatOutputs.exe | yourProgramToProcessInput.exe

在您的程序中,您可以使用 C 标准输入函数来读取其他程序的输出(scanf 等): http://irc.essex.ac.uk/www.iota-6.co.uk/c/c1_standard_input_and_output.asp 。您还可以使用文件示例并使用 fscanf。这应该也适用于 Unix/Linux。

这是一个非常通用的问题,您可能需要包含更多详细信息,例如它是什么类型的输出(只是文本,还是二进制文件?)以及您希望如何处理它。

编辑:澄清万岁!

重定向 STDOUT 看起来很麻烦,我不得不在 .NET 中这样做,这让我很头疼。看起来正确的 C 方式是生成一个子进程,获取一个文件指针,突然我头疼了。

这是一个使用临时文件的技巧。这很简单,但应该有效。如果速度不是问题(访问磁盘很慢),或者如果它是一次性的,那么这将很有效。如果您正在构建企业程序,那么使用其他人推荐的内容来研究 STDOUT 重定向可能是最好的选择。

#include <stdlib.h>
#include <stdio.h>

int main(int argc, char* argv[])
{
    FILE * fptr;                    // file holder
    char c;                         // char buffer


    system("dir >> temp.txt");      // call dir and put it's contents in a temp using redirects.
    fptr = fopen("temp.txt", "r");  // open said file for reading.
                                    // oh, and check for fptr being NULL.
    while(1){
        c = fgetc(fptr);
        if(c!= EOF)
            printf("%c", c);        // do what you need to.
        else
            break;                  // exit when you hit the end of the file.
    }
    fclose(fptr);                   // don't call this is fptr is NULL.  
    remove("temp.txt");             // clean up

    getchar();                      // stop so I can see if it worked.
}

确保检查您的文件权限:现在,这只会将文件放入与 exe 相同的目录中。您可能想考虑使用 /tmp 在尼克斯,或 C:\Users\username\Local Settings\Temp 在 Vista 中,或者 C:\Documents and Settings\username\Local Settings\Temp in 2K/XP. 。我觉得 /tmp 可以在 OSX 上运行,但我从未使用过。

在 Linux 和 OS X 中, popen() 正如 dmckee 指出的那样,这确实是您最好的选择,因为两个操作系统都支持该调用。在 Windows 中,这应该有帮助: http://msdn.microsoft.com/en-us/library/ms682499.aspx

MSDN文档说,如果在Windows程序中使用,则_POPEN函数将返回一个无效的文件指针,该指针导致该程序无限期停止响应。_popen 在控制台应用程序中正常工作。要创建重定向输入和输出的 Windows 应用程序,请参阅 Windows SDK 中的创建具有重定向输入和输出的子进程。

您可以使用 system() 如:

system("ls song > song.txt");

在哪里 ls 是列出文件夹歌曲内容的命令名称, song 是当前目录中的一个文件夹。结果文件 song.txt 将在当前目录中创建。

//execute external process and read exactly binary or text output
//can read image from Zip file for example
string run(const char* cmd){
    FILE* pipe = popen(cmd, "r");
    if (!pipe) return "ERROR";
    char buffer[262144];
    string data;
    string result;
    int dist=0;
    int size;
    //TIME_START
    while(!feof(pipe)) {
        size=(int)fread(buffer,1,262144, pipe); //cout<<buffer<<" size="<<size<<endl;
        data.resize(data.size()+size);
        memcpy(&data[dist],buffer,size);
        dist+=size;
    }
    //TIME_PRINT_
    pclose(pipe);
    return data;
}
许可以下: CC-BY-SA归因
不隶属于 StackOverflow
scroll top