先来介绍getchar这个函数的用法:
//getchar - 函数 - 从键盘读取字符的
#include<stdio.h>
int main()
{
int ch = 0;
while ((ch = getchar()) != EOF)
{
putchar(ch);
}
return 0;
}
可以看到运行结果:输入和输出是一致的。
细心的你发现问题了吗?第一次输入‘A’时,打印‘A’后光标自动换行了!下面的‘X’和‘C’也是这样,这是巧合吗?
别急,再来看看下面的代码:
#include<stdio.h>
int main()
{
char password[20] = { 0 };
printf("请输入密码:");
scanf("%s", password);
printf("请确认密码(Y/N):");
int ch = getchar();
if (ch == 'Y')
{
printf("确认成功\n");
}
else
{
printf("确认失败\n");
}
return 0;
}
运行结果:
从这里可以看到,我在输入密码后,还没有确认密码,第二行就直接显示确认失败。这是什么原因导致的?
我们通过键盘输入密码,scanf/getchar并不是直接获取输入的字符,而是在键盘和scanf/getchar之间有一个缓冲区,第一次输入后,缓冲区里面存放的内容是:abcdef\n。
scanf首先读走了abcdef,scanf不读取\n。此时缓冲区只剩下\n了,下一次getchar发现缓冲区里面还有一个\n,于是就直接读取了\n,由于\n!=‘Y’,此时走了else语句,打印出了确认失败。
用图来表示的话:
既然我们知道了问题所在,那我们该如何解决问题呢?也就是有效消除‘\n’。
第一次改进:
#include<stdio.h>
int main()
{
char password[20] = { 0 };
printf("请输入密码:");
scanf("%s", password);
getchar();
printf("请确认密码(Y/N):");
int ch = getchar();
if (ch == 'Y')
{
printf("确认成功\n");
}
else
{
printf("确认失败\n");
}
return 0;
}
运行后结果:
但是这样以为就结束了吗?如果是这样的情况呢?
在这里可以看到,如果在输入过程中1存在空格字符,我们发现再次出现我们还没有确认密码就出现了确认失败。
原因:scanf不会读取空格字符,在读取abc后就停止读取了,getchar也不会读取空格字符,于是往后自动读取了字符‘d’, 于是就确认失败了。
有人在这里想,再多写几个getchar接收空白字符不就行了?这种方法显得有点笨拙。我们这样来造。
#include<stdio.h>
int main()
{
char password[20] = { 0 };
printf("请输入密码:");
scanf("%s", password);
int tmp = 0;
//清理缓冲区里面的内容
while ((tmp = getchar()) != '\n')
{
;
}
printf("请确认密码(Y/N):");
int ch = getchar();
if (ch == 'Y')
{
printf("确认成功\n");
}
else
{
printf("确认失败\n");
}
return 0;
}
//清理缓冲区里面的内容
while ((tmp = getchar()) != ‘\n’)
{
;
}这里意思是,scanf读取字符后,getchar开始读取,直到读取到‘\n’后才停止读取,在此期间进入while循环内部,但是不执行命令。这样一来就有效解决了缓冲区内多余字符的问题。(缓冲区可以留有空格,但是不会影响命令执行,不参与其中。)
运行结果:
注意:关于缓冲区的问题是对于输入字符类型而言的,对于整型等其他类型不用考虑。