stderr,stdin,stdout相关

转载请保留原作者。

一.stdin和stdout

1.意义

stdin表示标准输入,stdout表示标准的输出。它们都是文件指针。默认情况下,所指向的文件是控制台。
可以使用freopen进行重定向,或者fopen新建文件指针,都是可以的。

2.缓冲

2.1.scanf的缓冲问题

#include<stdio.h>
using namespace std;
int main(){
    int a;
    while(scanf("%d",&a)!=1)printf("error!");
    return 0;
}

这段代码的原意是,如果读入不到a,那么就输出error。
但是,stdin中的内容,scanf不是按照行读入字符,而是根据连续的字符流读入字符。因此,如果读入一个非法字符,例如非数字的字符,那么会出问题。因为读入后,scanf发现错误,然后输入error,再次读入下一个字符时,上次的那个字符还残留在流中,因此导致无限输出error。

对于这个做法,有人会使用fflush(stdin),这个问题我们下一章节继续。

2.2.fflush

一般来说,stdout是有缓冲的。缓冲的意思就是先把字符存起来,到达某一时刻,把缓存的字符一次性写入文件。我们知道,硬盘速度比内存慢很多,每写一个就存入一次速度很慢,如果累计起来一次放入速度会提高。
stdout默认是到达换行符刷新。

fflush一般用于输出的缓冲刷新。也就是说,输出到一半,使用fflush可以提前刷新。
但是fflush只能用于输出的刷新,输入流不能这样刷新。fflush(stdin)可能在部分编译器可以刷新输入流,达到上文的scanf刷新,但是标准没有定义这个行为。也就是说,部分编译器是不支持的。

3.freopen

freopen(const char *filename,const char *mode,FILE *stream);
把stream重定向到filename的文件名去。
例如,freopen(“a.txt”,”r”,stdin);表示把a.txt作为输入重定向stdin。这样,默认scanf就会从a.txt读入而不是控制台。
第二个参数mode,表示模式,例如r表示文本读入,w表示文本写入,rb表示二进制读入等。

如果在重定向到文件后,重定向回控制台进行输入输出,可以把filename设置为“CON”即可。

二.stderr

1.输出方法

标准错误流。fprintf(stderr,...);
使用fprintf指定文件指针为stderr,进行输出,默认输出文件还是stdout的文件。也就是说,也可以往控制台输出,输出位置一样的。

2.默认缓冲

让我们写个程序检验一下。

#include<stdio.h>
using namespace std;
int main(){
    fprintf(stdout,"stdout ");
    fprintf(stderr,"stderr ");
}

输出:stderr stdout
看上去stderr先输出?没错,就是这样,没有换行符的时候,stdout不会自动刷新,但是stderr每写一个字符都要刷新。因此,stderr先输出,程序结束的时候再刷新一次,输出stdout。

如果先加上fflush,那么就是stdout先了。

#include<stdio.h>
using namespace std;
int main(){
    fprintf(stdout,"stdout ");fflush(stdout);
    fprintf(stderr,"stderr ");
}
posted @ 2021-09-05 20:16  计算机知识杂谈  阅读(477)  评论(0编辑  收藏  举报