C语言关于缓冲区的问题

  • Post author:
  • Post category:其他


先来介绍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循环内部,但是不执行命令。这样一来就有效解决了缓冲区内多余字符的问题。(缓冲区可以留有空格,但是不会影响命令执行,不参与其中。)

运行结果:

注意:关于缓冲区的问题是对于输入字符类型而言的,对于整型等其他类型不用考虑。



版权声明:本文为qq_58724706原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。