getchar()
首先就是查这个问题的应该是对函数的功能有一定的了解的,简单来说就是从标准输入流stdin读取一个字符。
它可以等价与int getc(FILE *stream)函数在stdin上的实现,此处只是等价。
发现问题
场景:windows + clion编译器,如果环境一样的话可以测试一下如下代码:
#include <stdio.h>
int main(){
int a, b, c, d;
printf("Please input:");
a = getchar();
b = getchar();
c = getchar();
d = getchar();
printf("%d %d %d %d", a, b, c, d);
}
执行代码之后,可以得到如下结果:
- 我们输入了四个字符,分别为a、\n、b、\n,根据结果可以看到,a取值为97也就是字符‘a’,b取值为10也就是换行符的ASCII值,我们会惊讶的发现c取值也为换行符的ASCII值,d取值为98也就是字符b的ASCII值。
因为getchar就是从输入缓冲区读取字符所以前两个变量的取值结果并不意外,按照正常的思维来说,c变量的取值应该是字符‘b’,d的取值应该是换行符。先不着急去寻找为什么会产生这种原因,换个系统试试,这里我又选取了Linux系统进行测试,应用上述相同的代码可以得到如下结果:
截图有点小,水印盖了一大半,大家凑合看一下(哭)。
从结果可以看到,和windows系统上的并不太一样,linux上的结果和我们正常思考的结果一样,那为什么会造成这种原因呢,我们可以暂时把linux系统上的结果作为一种标准答案,以此来探究为什么windows上会出现这种情况。
探究过程
- 猜想1:由于缓冲区处于I/O操作和与数据流相关的文件之间,输入流stdin将数据暂存到缓冲区中,因此会不会是缓冲区添加了新的换行符,造成了这个问题?
这里我们要明确几个概念:
(1)缓冲区是什么?
缓冲区是一种中间存储区域,数据的输入和输出都可以先经过缓冲区,再发送到屏幕,缓冲区提供了一种容错机制,也就是用户输入错误的时候,可以用退格键删除曾经输入的字符。
(2)缓冲区什么时候发送数据?
当缓冲区满、遇到换行符或者需要输入的时候。这里也涉及到完全缓冲I/O、行缓冲I/O和无缓冲三种模式,具体概念大家可以搜搜,很容易找到。把缓冲区数据发送到屏幕或者文件被称为刷新缓冲区。
键盘输入一般为行缓冲,所以按下Enter键后才刷新缓冲区。
#include <stdio.h>
int main(){
char buf[5] = {
0};
char arr[5] = {
0};
setvbuf(stdin, arr, _IOLBF, 5);//设置stdin的缓冲区
printf("Please input:");
fread(buf, 1, 2, stdin); //从标准输入流读取两个字符
for(int i = 0