C printf 无法从终端按预期工作

C printf not working as expected from terminal

本文关键字:工作 终端 printf      更新时间:2023-10-16

我已经尽可能地将问题简化为以下最小的代码片段。如果需要更多信息,请询问。

假设我有一个名为 A 的 Java 类,其主要方法是这样的:

public static void main (String args[])throws Exception{
Process p = new ProcessBuilder(args).redirectErrorStream(true).start();
Scanner in = new Scanner(new InputStreamReader(p.getInputStream())); 
System.out.println(in.nextLine());
}

基本上,它从终端运行一个程序,从该程序读取一行,并回显它。很简单。

现在我有第二个用 c++ 编写的程序,如下所示:

int main() 
{
string s;
cout << "example2 in c++"<<endl;;
cin >> s;
return 0;
}

当我调用:

java A pathToC++Program

程序按预期运行,输出从 CPP 程序发送的字符串并终止。

我在 c 中有一个等效的程序:

int main()
{
char s[10];
printf("example1 in cn");
fgets(s,10,stdin); 
return 0;
}

当我调用:

java A pathToCprogram

程序停止,并且未打印任何内容。

当我从 c 程序中删除 fgets 行时,它看起来像这样:

int main()
{
printf("example1 in cn");
return 0;
}

然后Java程序能够识别该行并将其打印出来。因此,fgets 似乎是有问题的行,但是当包含 fgets 时,即使是之前的行也不起作用。

这是怎么回事?

(我不知道这是Java还是C的特点,所以我包含了这两个标签。

写入stdout(这是printf写入其输出的位置(默认是行缓冲的,这意味着缓冲区将在换行符上刷新。

这是直接从终端运行时的默认设置。如果输出不是终端,则stdout将被完全缓冲,这意味着只有在填充缓冲区时才会刷新它。或显式fflush(stdout)调用。

在 C++ 程序中,您可以使用std::endl显式刷新输出(它写入换行符并刷新缓冲输出(。在 C 程序中,输出缓冲区没有这样的刷新。

这确实是非常特定于平台的,但对于POSIX平台(如Linux或macOS(,它就像上面描述的那样。我认为这与Windows非常相似。

至于为什么当你删除fgets语句时它会起作用,可能是因为fgets阻止执行,直到读取文件末尾或某些(换行符终止的(输入。如果没有fgets程序将在printf后立即退出,这将导致stdout缓冲区被刷新。

std::endl传递给std::coutC++将刷新输出缓冲区。

若要刷新 C 中的输出缓冲区,应在printf行后添加fflush(stdout);