gpt4 book ai didi

c - 在C中使用管道后,如果我将stdout重定向到文件,该文件包含乱码

转载 作者:行者123 更新时间:2023-11-30 17:36:47 25 4
gpt4 key购买 nike

我是在 C 代码中使用管道的新手,所以我遇到一些令人困惑的情况需要您的帮助,

我需要使用opt -print-callgraph foo.bc来输出函数调用图。

默认情况下,调用图将位于 stderr 中,我可以使用 opt -print-callgraph foo.bc 2>call_graph.txt 将 stderr 重定向到文件。

我需要使用 C 程序,给出 foo.bc 作为参数并生成 call_graph.txt

假设编译后的二进制文件名为prog,运行./prog foo.bc后就可以正确获取call_graph.txt。

但我的问题是,如果我将 prog 的标准输出重定向到一个文件,也就是prog foo.bc > output.txt,output.txt包含大量乱码。 output.txt 可能是一个二进制文件。

我不知道管道在代码中做什么,或者我应该在哪里清理标准输出。

这是我的代码:

int main(int argc, char *argv[]){   

/* pipe to connect opt's stderr and our stdin */
int pipe_callgraph[2];

/* pid of opt */
int opt_pid;
char *bc_file;

/* create pipe and check if pipe succeeded */
if (pipe(pipe_callgraph) < 0) {
perror("pipe");
return 1;
}

/* create child process */
opt_pid = fork();
if (!opt_pid) { /* child process, to spawn opt */

/* close the read end, since opt only write */
close(pipe_callgraph[PIPE_READ]);

/* bind pipe to stderr, and check */
if (dup2(pipe_callgraph[PIPE_WRITE], STDERR_FILENO) < 0) {
perror("dup2 pipe_callgraph");
return 1;
}

/* print something to stderr */
fprintf(stderr, "This is child, just before spawning opt with %s.\n", bc_file);

/* spawn opt */
if (execl("/usr/local/bin/opt", "opt", "-print-callgraph", bc_file, (char *)NULL) < 0) {
perror("execl opt");
return 1;
}

/* unreachable code */
return 0;
}

/* parent process */

/* close the write end, since we only read */
close(pipe_callgraph[PIPE_WRITE]);

/* since we don't need stdin, we simply replace stdin with the pipe */
if (dup2(pipe_callgraph[PIPE_READ], STDIN_FILENO) < 0) {
perror("dup2 pipe_callgraph");
return 1;
}

FILE *fp = fopen("call_graph.txt", "w");
char c = '\0';
while (scanf("%c", &c) >= 1) {
fprintf(fp, "%c", c);
}
fclose(fp);

输出.txt如下所示:

enter image description here

==============编辑================

如果我更改这些代码

FILE *fp = fopen("call_graph.txt", "w");
char c = '\0';
while (scanf("%c", &c) >= 1) {
fprintf(fp, "%c", c);
}
fclose(fp);

FILE *fp = fopen("call_graph.txt", "w");
fclose(fp);

output.txt仍然存在乱码

如果我注释掉这两行,就不会有乱码了。

如果像这样改变:

fflush(stdout);
FILE *fp = fopen("call_graph.txt", "w");
fclose(fp);

无乱码。

所以我以这种方式多次刷新标准输出:

fflush(stdout);
FILE *fp = fopen("call_graph.txt", "w");
char c = getchar();
while (c >= 1) {
fflush(stdout);
fprintf(fp, "%c", c);
fflush(stdout);
c = getchar();
}
fflush(stdout);
fclose(fp);

output.txt中仍然有乱码

最佳答案

我发现了问题,那是因为 opt -print-callgraph foo.bc 将调用图打印到 stderr 并将 foo.bc 的内容打印到 stdout,所以我需要使用 >/dev/null 捕获标准输出。

乱码与管道无关。

关于c - 在C中使用管道后,如果我将stdout重定向到文件,该文件包含乱码,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/22573565/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com