c语⾔缓冲区类型,清空缓冲区,谈getchar、getch、getche
C语⾔缓冲区(缓存)详解
缓冲区(Buffer)⼜称为缓存(Cache),是内存空间的⼀部分。也就是说,在内存中
预留了⼀定的存储空间,⽤来暂时保存输⼊或输出的数据,这部分预留的空间就叫做缓冲
区。 缓冲区根据其对应的是输⼊设备还是输出设备,分为输⼊缓冲区和输出缓冲区。
为什么要引⼊缓冲区
⽐如从磁盘⾥取信息,我们先把读出的数据放在缓冲区,计算机再直接从缓冲区中取数
据,等缓冲区的数据取完后再去磁盘中读取,这样就可以减少磁盘的读写次数,再加上计算
机对缓冲区的操作⼤⼤快于对磁盘的操作,故应⽤缓冲区可⼤⼤提⾼计算机的运⾏速度。
⼜⽐如,我们使⽤打印机打印⽂档,由于打印机的打印速度相对较慢,我们先把⽂档输
出到打印机相应的缓冲区,打印机再⾃⾏逐步打印,这时我们的CPU可以处理别的事情。
现在你基本明⽩了吧,缓冲区就是⼀块内存区,它⽤在输⼊输出设备和CPU之间,⽤
来缓存数据。它使得低速的输⼊输出设备和⾼速的CPU能够协调⼯作,避免低速的输⼊输
出设备占⽤ CPU,解放出 CPU,使其能够⾼效率⼯作。
缓冲区的类型
缓冲区分为三种类型:全缓冲、⾏缓冲和不带缓冲。
1) 全缓冲
在这种情况下,当填满缓冲区后才进⾏实际 I/O操作。全缓冲的典型代表是对磁盘⽂件
的读写。
2) ⾏缓冲
在这种情况下,当在输⼊和输出中遇到换⾏符时,执⾏真正的 I/O操作。这时,我们输
⼊的字符先存放在缓冲区,等按下回车键换⾏时才进⾏实际的I/O操作。典型代表是标准输
⼊(stdin)和标准输出(stdout)。
3) 不带缓冲
也就是不进⾏缓冲,标准错误⽂件 stderr 是典型代表,这使得出错信息可以直接尽快
地显⽰出来。
C语⾔清空缓冲区
scanf() 的缓冲区有时会引发奇怪的问题,多个 scanf() 之间要注意清空缓冲区。清空缓冲区主要有两种思路:⼀是将缓冲区中的数据丢弃,⼆是将缓冲区中的数据读取出来,但是却不使⽤。
fflush(stdin)
fflush() 函数⽤来清空⽂件缓冲区,它的原型为:
int fflush(FILE *stream)
stream 为流指针,可以理解问⼀个⽂件指针。在 C语⾔中,为了便于操作,键盘和显
⽰器也被看作是⽂件,这样对硬件的操作就等同于对⽂件的操作。键盘称为标准输⼊⽂件(stdin),显⽰器称为标准输出⽂件(stdout)。
如此就可以使⽤ fflush() 来清空输⼊缓冲区中的数据,具体⽤法为:
fflush(stdin);
请看下⾯的代码:
运⾏结果:
100 200↙
300↙
a=100, b=300
第⼀个 scanf() 读取完成后,将 100 赋值给变量 a,缓冲区中剩下 200。然后调⽤
fflush() 函数将 200 从缓冲区中清除。执⾏到第⼆个 scanf() 时由于缓冲区中没有数据,
所以会等待⽤户输⼊,将 300 赋值给变量 b。
如果把第 7 ⾏代码注释掉,运⾏结果为:
100 200↙
a=100, b=200
由于没有清空缓冲区,执⾏到第⼆个 scanf() 时直接将缓冲区中的 200 赋值给变量 b。
fflush(stdin) 直接将缓冲区中的数据丢弃,是初学者常⽤的清空输⼊缓冲区的⽅法,它
在 Windows 下⼀般是有效的,但在 Linux GCC 下可能⽆效,因为 C 语⾔标准规定:对
于以 stdin 为参数的 fflush() 函数,它的⾏为是不确定的,fflush() 操作输⼊流是对标准
c语言char的用法
C语⾔的扩充。
循环读取缓冲区中的数据
从缓冲区中读取剩余数据的⽅法也很多,这⾥讲解常⽤的两种。
使⽤ getchar() 读取数据:
int c;
while((c = getchar()) != '\n' && c != EOF);
该代码不停地使⽤ getchar() 获取缓冲区中的字符,直到获取的字符是换⾏符\n 或者
是⽂件结尾符 EOF 为⽌。这个⽅法可以完美清空输⼊缓冲区,并且具备可移植性。
使⽤ scanf() 读取:
scanf("%*[^\n]%*c");
%*[^\n]将逐个读取缓冲区中的 ‘\n’ 字符之前的其它字符,% 后⾯的 * 表⽰将读取
的这些字符丢弃,遇到 ‘\n’ 字符时便停⽌读取。此时,缓冲区中尚有⼀个 ‘\n’ 字符遗留,所以后⾯的%*c 将读取并丢弃这个遗留的换⾏符,这⾥的星号和前⾯的星号作⽤相同。由于所有从键盘的输⼊都是以回车结束的,⽽回车会产⽣⼀个 ‘\n’ 字符,所以将 ‘\n’连同它之前的字符全部读取并丢弃之后,也就相当于清除了输⼊缓冲区。
请看下⾯的例⼦:
运⾏结果:
100 200↙
300↙
a=100, b=300
9 99↙
999↙
a=9, b=999
虽然两种⽅法都能起到清空缓冲区的作⽤,但第⼀种⽅法需要额外定义⼀个 char 类
型的变量,略显繁琐,并且 while 循环也会导致效率不⾼,所以建议使⽤第⼆种⽅法。
综上所述:如果只考虑 Windows,建议使⽤ fflush(stdin);,简单明了;如果兼顾移植和效率,建议使⽤ scanf("%*[^\n]%*c");,虽然有点蹩脚,但确实能够奏效。
结合C语⾔缓冲区谈getchar()、getche()、getch()
getchar()、getche()、getch() 函数,它们都⽤来从控制台获取字符,getchar() 会等
待⽤户按下回车键才开始读取,⽽ getche()、getch() 会⽴即读取。这是因为 getchar() 带有缓冲区,⽤户输⼊的数据会暂时保存到缓冲区,直到按下回车键才开始读取;⽽ getche()、getch() 不带缓冲区,只能⽴即读取。
getchar()函数
getchar() 函数的特点是:如果缓冲区中没有内容,那么等待⽤户输⼊;如果有内容,
哪怕⼀个字符,也会直接从缓冲区中读取数据,不会等待⽤户输⼊。
第⼀次调⽤ getchar() 时,会等待⽤户输⼊,⽤户输⼊的所有字符都被放到标准输⼊
(stdin)缓冲区,直到⽤户按下回车键为⽌(回车符也被放⼊缓冲区)。⽤户按下回车键,
getchar() 函数才开始从缓冲区中读取数据,每次读取⼀个字符。
下⾯我们借助 getchar() 函数将 \n 从缓冲区中清除:
#include <stdio.h>
#include <stdlib.h>
int main()
{
int a=0, b=0;
scanf("a=%d", &a);
getchar();
scanf("b=%d", &b);
printf("a=%d, b=%d\n", a, b);
system("pause");
return 0;
}
运⾏结果:
a=100↙
b=100↙
a=100, b=100
执⾏完第⼀个 scanf() 后,缓冲区中剩下换⾏符 \n,我们使⽤ getchar() 将其读出(并
不使⽤),执⾏到第⼆个 scanf() 时,由于缓冲区中没有内容,所以会等待⽤户输⼊。
getch()函数
getch 和 getchar 的作⽤类似,都是从键盘读取⼀个字符,但是:
getch 不带回显,也就是说,你输⼊的字符不会在屏幕上显⽰出来。
getch 没有缓冲区,也就是说,输⼊⼀个字符就⽴即读取。
getch 使⽤举例。
#include <stdio.h>
#include <conio.h>
int main(){
char c1, c2;
c1 = getch();
printf("%c\n", c1);
c2 = getch();
printf("%c\n", c2);
return 0;
}
先输⼊ 'a',再输⼊ 'b',运⾏结果为:
a
b
输⼊⼀个字符,getch 会⽴即获取,不会给你多输⼊⼀个字符的机会。并且输⼊的字
符只由 printf 语句显⽰⼀次,getch 不会显⽰。
注意要包含头⽂件 conio.h,getch 和 getche 都在该头⽂件中声明。
⼀般情况下,程序运⾏结束后要暂停⼀下才能看到输出结果,否则只能看到⼀个“⿊影”
⼀闪⽽过,所以要在程序最后添加 system(“PAUSE” );语句,如果使⽤ C-Free 或 VC 6.0运⾏程序,会⾃动添加该语句。system(“PAUSE”);语句会输出⼀⾏多余的⽂字,如果你
不会喜欢这样,也可以⽤ getch 函数来实现“暂停”的效果。请看下⾯的代码:
#include <stdio.h>
#include <conio.h>
int main(){
printf("%s", "getch is great!");
getch();
return 0;
}
运⾏程序,输出字符串 getch is great!后,按任意键程序就会结束。
使⽤ getch 的好处是,不管你按什么键,都不会在屏幕上留下痕迹,使你的界⾯达到美观效果。getche()函数
getche()和 getch()很相似,也没有缓冲区,区别在于:getch()⽆回显,getche()有回
显。
#include <stdio.h>
#include <conio.h>
int main(){
char c1, c2;
c1 = getche();
printf("%c\n", c1);
c2 = getche();
printf("%c\n", c2);
return 0;
}
先输⼊ 'a',再输⼊ 'b',运⾏结果为:
aa
bb

版权声明:本站内容均来自互联网,仅供演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系QQ:729038198,我们将在24小时内删除。