C语言scanf函数输入时键盘缓冲区\n的问题

你好!我想请教你关于利用scanf函数输入时,留在键盘缓冲区的\n(ASCII值为10)的问题。期待你的回答。至于分数,你想要多少,说就行。我昨晚写程序时对scanf在键盘缓冲区留下的字符有疑问,思考不果。看了百度百科上的scanf词条,说scanf输入遇到空格、跳格、回车才会从缓冲区往变量送字符。于是自己写了以下几个程序思考,还是不果。

程序1
#include "stdio.h"

void main()
{
char a;
char b;
scanf("%d",&a);
scanf("%d",&b);
printf("%d %d",a,b);
}

键盘输入
97<回车>96<回车>

输出
97 96

问题1:调用第一个scanf输入时,键盘缓冲区所有的字符为97\n,遇到回车,所以缓冲区把97赋值给a。调用第二个scanf输入时,键盘缓冲区所有的字符为96\n,遇到回车,所以缓冲区把96赋值给b。以上我的分析对吗?

程序2
#include "stdio.h"

void main()
{
char a;
char b;
scanf("%c",&a);
scanf("%c",&b);
printf("%d %d",a,b);
}

键盘输入
9<回车>

输出
57 10

问题2:调用第一个scanf输入时,键盘缓冲区所有的字符为9\n,遇到回车,所以缓冲区把9赋值给a。调用第二个scanf输入时,键盘缓冲区所有的字符为\n,遇到回车,所以缓冲区把\n赋值给b。以上我的分析对吗?如果对,那程序1中调用第一个scanf时,又为什么不是把97赋值给a后,将\n赋值给b呢?为什么调用第二个scanf时还需要继续输入96<回车>来对b赋值?调用第一个scanf输入时留在缓冲区的\n去哪里了?无端消失了?

程序3
#include "stdio.h"

void main()
{
char a[100];
char b[100];
scanf("%s",a);
scanf("%s",b);
printf("%s %s",a,b);
}

键盘输入
abc<回车>def<回车>

输出
abc def

问题3:从输出结果可以看出,字符数组a和字符数组b都在同一行输出。所以字符数组a的值为{‘a’,’b’,’c’},不是{‘a’,’b’,’c’,’\n’}。字符数组b的值为{‘d’,’e’,’f’},不是{‘\n’,‘d’,’e’,’f’},也不是{‘\n’,‘d’,’e’,’f’,’\n’}。以上的分析对吗?如果对,那调用第一个scanf输入时留在缓冲区的\n去哪里了?还有第二个scanf留下\n呢?

程序4
#include <stdio.h>
void main()
{
int i;
char j;
for(i=0;i<2;i++)
scanf("%c",&j);/*注意这里%前没有空格*/
printf("%d",j);
}

键盘输入
1<回车>

输出
10

程序5
#include <stdio.h>
void main()
{
int i;
char j;
for(i=0;i<2;i++)
scanf(" %c",&j);/*注意这里%前有一个空格*/
printf("%d",j);
}

问题4:程序4应该就像程序2那样,最后把\n(ASCII值为10)赋值给j了,所以输出10。但程序5 scanf里那个空格如何阻止\n给j赋值?想不通,恳请赐教!

程序6
#include "stdio.h"

void main()
{
int a;
int b;
scanf("%c",&a);
scanf("%c",&b);
printf("%d %d",a,b);
}

键盘输入
1<回车>

输出
-858993615 -858993654

问题5:这个问题和\n无关的,但写了程序1,却发现了这个问题。我用的是VC6,就算不是VC6,C的任何编译软件里int和char不都是通用的吗?为什么程序1用%d格式能正常获得char型变量,但程序6用%c格式却不能正常获得int型变量?

我知道我问得的确很多,但非常期待您的回答!想要多少分数,你开口就行,我有多少就给多少,反正能学到东西才是重要的!感谢!!!

你首先要明白,从键盘读入键盘缓冲区(buffer)的数据都是以ASCII码存储的(包括回车)。
程序1
#include "stdio.h"

void main()
{
char a;
char b;
scanf("%d",&a);
scanf("%d",&b);
printf("%d %d",a,b);
}

键盘输入
97<回车>
第一次回车后,buffer中的ASCII:39h,37h,0AH(0A是换行的ASCII), scanf会根据格式字符串中的第一个%d对buffer按字节顺序读取,当读取到0A时,认为%d型的数据结束,此时把已经读取到的39h,37h依据%d转为整型数据97存储在字符型变量a中。(这里是除去了扫描截止点0AH)
此时buffer中已经无任何数据了。

96<回车>
第二次回车后,按同样的流程,scanf会根据格式字符串中的第二个%d对buffer按字节顺序读取。最终b得到96.
此时buffer中已经无任何数据了。
输出
97 96

程序2
#include "stdio.h"

void main()
{
char a;
char b;
scanf("%c",&a);
scanf("%c",&b);
printf("%d %d",a,b);
}

键盘输入
9<回车>buffer:39H,0AH
因为scanf会按照第一个%c格式扫描buffer(只扫描一个字节就结束),然后把扫描到的39H直接送到变量a(当以%d格式读出来时,39H就是57)
此时,buffer中只有:0AH。
然后,scanft又遇到第二个%c,继续扫描buffer,得到0aH并送入变量b.
此时buffer中已经无任何数据了

输出
57 10

程序3
#include "stdio.h"

void main()
{
char a[100];
char b[100];
scanf("%s",a);
scanf("%s",b);
printf("%s %s",a,b);
}

键盘输入
abc<回车>
第一次回车后,buffer:61H,62H,63H,0AH。
scanf会按照%s的格式对buffer按字节顺序扫描,当扫描到0AH时,结束扫描(按照%s的要求,空格20H也是扫描结束点)。
然后把扫描到的(除去最后一个判断扫描截至的字节0AH)数据直接送入以a为起始地址的字符串。
此时,buffer无任何数据了。

def<回车>
第二次回车后,buffer:65H,66H,67H,0AH.扫描的流程与上面的完全一致。
输出
abc def

程序4
#include <stdio.h>
void main()
{
int i;
char j;
for(i=0;i<2;i++)
scanf("%c",&j);/*注意这里%前没有空格*/
printf("%d",j);
}

键盘输入
1<回车>,
这里scanf执行了两次(i==0时,与i==1时),而且每次都是想对j赋值。
第一次scanf,按%c的要求,只扫描buffer中的一个字节,但是buffer中并不数据,于是要求键盘输入数据到buffer,此时的1<回车>代表向buffer中输入了:31H,0AH。
然后按%c的要求,只扫描buffer中的一个字节:31h,并将它直接送入变量j.
此时,buffer中还留下:0AH。

第二次scanf要求键盘输入数据,按%c的要求,只扫描buffer中的一个字节:0Ah,并将它直接送入变量j.
此时,buffer无数据了。

最后,你用%d格式输出j的值(0AH换成整型就是10)

输出
10

程序5
#include <stdio.h>
void main()
{
int i;
char j;
for(i=0;i<2;i++)
scanf(" %c",&j);/*注意这里%前有一个空格*/
printf("%d",j);
}
1<回车>2<enter>的情况:
scanf会按照格式控制字符串的要求,顺序扫描buffer.
但是你其中有一个空格,这个很特殊,我也是第一次发现这个问题(一般我都不会在scanf中加入任何常量字符)

我测试了一下:我发现这个空格有吸收回车(0AH)和空格(20H)的“神奇功效”,吸收之后再要求buffer给一个字节,直到这个字节不是0AH或者 20H,此时把这个字节交给下一个格式字串。

第一次循环时遇到格式字串空格,就扫描buffer中的一个字节,但是buffer中无数据,要求从键盘输入数据:1〈回车〉,buffer中有数据了——31H,0AH。再读取到字节31H,scanf发现这个并不是0AH/20H,就把这个字节31H交给格式字符%c处理。
循环结束,此时buffer里面还有:0AH.

第二次循环时遇到格式字串空格,就扫描buffer中的一个字节——0AH,发现是0AH/20H,于是就要求buffer再来一个字节。此时buffer里面已经没有数据了,要求键盘输入:2<enter>.
buffer中有数据了——32H,0AH。于是再读一个字节31H,scanf发现这个并不是0AH/20H,就把这个字节32H交给格式字符%c处理(j最终得到32H)。
循环结束,此时buffer里面还有:0AH.

这里有一篇关于Printf的帖子:http://blog.csdn.net/arong1234/archive/2008/05/18/2456455.aspx

程序6
#include "stdio.h"

void main()
{
int a;
int b;
scanf("%c",&a);
scanf("%c",&b);
printf("%d %d",a,b);
}

键盘输入
1<回车>

问题5:

你的编译器VC认为%d数据应该是4个字节,但是你采用的是%c读数据,
scanf("%c",&a);此句读到的是1的ascii码:31h.然后把31H直接送入地址&a(而并没有改写a的三个高字节地址)。
scanf("%c",&b);同理。
你可以用printf("a=%x,b=%x\n",a,b);来验证我说的。它们的最低字节肯定是31H,0AH。

PS1:
当你把 int a;int b;放在main()外进行定义时,a,b的初值就是0。此时你会得到正确的结果。
当你把 int a;int b;放在main()内进行定义时,a,b不会被初始化(它们的三个三个高字节地址的内容是不确定的),你就会得到上面错误的结果。(定义的动态变量都不会被初始化,静态变量会被初始化为0)

PS2:以下也是不正确的用法。
char c;
scanf("%d",&c);/当你用%d给c赋值时,会对从&c开始的连续4个字节进行赋值。当从buffer得到的值是在一个字节范围内(-128~127),下面是可以正常输出的。但是不管怎样,这样做是很危险的——越界。
printf("%d",c);
=================请你测试下这个程序========================
#include "stdio.h"
void main()
{
char c[4],i=4;
scanf("%d",c);/*请输入258<回车>*/

while(i-->0)
printf("%02x ",c[i]);
printf("\n");
}/*如果得到的结果是00 00 00 01 02就说明我的结论是正确的(258的转为16进制数就是00 00 01 02H,然后scanf会把这个数放入以c为起始地址的)

================以下程序也是======================
#include "stdio.h"
void main()
{
char c,i=4;
char *p=&c;
scanf("%d",&c);/*请输入258<回车>*/

while(i-->0)
printf("%02x ",p[i]);
printf("\n");
}
温馨提示:内容为网友见解,仅供参考
第1个回答  2010-04-04
你忽略了一个问题,就是scanf的格式控制符的用处。%d只接受整数的输入,%c和%s可以接受字符的输入,其中也包括\n等转义字符。因此“scanf输入遇到空格、跳格、回车才会从缓冲区往变量送字符”不是绝对的。清楚了%c %s可以接受字符,问题就解决了。
至于程序6,我不用VC不清楚,但gcc是可以通过的
第2个回答  2010-04-04
getchar 和 %c 格式 涉及 '\n',常使初学者困扰。
清除缓冲区残存信息其实很简单:
fflush(stdin);
就可以了,这样,后面的读就不受前面的影响。

键入数据的格式应当与输入语句格式相配:
char a; 相当1字节整型。所以可以用%d格式输入。用%d格式,默认分隔符是所有的 white-spaces

用%c格式输入,则按ASCII字符考虑。无分隔符。必要时用fflush(stdin);清除缓冲区

%s 是 字符串格式,默认分隔符是所有的 white-spaces,输入后自动有结束符"\0".

C语言scanf函数输入时键盘缓冲区\\n的问题
scanf("%c",&j);\/*注意这里%前没有空格*\/ printf("%d",j);} 键盘输入 1<回车>,这里scanf执行了两次(i==0时,与i==1时),而且每次都是想对j赋值。第一次scanf,按%c的要求,只扫描buffer中的一个字节,但是buffer中并不数据,于是要求键盘输入数据到buffer,此时的1<回车>代表向buffer...

关于C语言scanf中\\n 的问题
scanf("%d\\n", &i);printf("%d\\n", i);你输入一个数后,它是不会立即显示的,要等再接收到一个非(空格、制表符、回车)的输入scanf语句才结束。但请注意的是,最后输入的那个非(空格、制表符、回车)的东西是不会被这个scanf读进来的,而是留在输入流里。所以3会被x读入,也就是说最后...

为什么scanf那一行语句加了\\n得输入两次字符才能输出,去掉\\n就正常了...
scanf()函数是格式化输入函数,它从标准输入设备(键盘) 读取输入的信息。其调用格式为: scanf("<格式化字符串>",<地址表>);解释原因:scanf在每次回车之后从缓冲区中读取用户输入的字符,且当缓冲区非空时,输入回车之后才会自动结束。当按照scanf("%c\\n", &m)这种形式书写时,相当于你输入结...

谁帮我看下 C语言问题?scanf中加\\n之后的作用
当你从键盘输入数据时,是先放在该缓冲中,然后scanf才从该缓冲中读数据,你不加\\n时,你输入i的值,然后回车想再输入m的值吧?这时送到缓冲中的就是i的值和一个回车符,所以scanf第一次读的是i,第二次读的就是一个回车符,所以printf(" %c\\n",m);输出的是回车,而不是你理想中的m值,...

c语言关于scanf的小问题
解决的办法就是在输入字符前,用一条语句把那个遗留在缓冲区中的‘\\n'给接收了或删除了就行,用getchar()是接收,用fflush(stdin)是删除,都是可以的。而在%c前加一个空格和getchar()作用一样,因为scanf()函数中的控制符中加入什么符号就要输入什么符号,这个空格就相当于要求输入一个空格,刚好把...

c语言新手scanf问题
是scanf("%f\\n",&fSalary)和scanf("%d\\n",&iNum)两条语句格式化输入中“\\n”造成的。预备知识:(1)scanf()函数接收输入数据时,遇回车键,开始从键盘缓冲区读输入字符(这个回车符不被读入,并留在缓冲区中),并按格式化输入指定条件,赋给相应变量;读入字符时,有非空字符(空字符如:空格...

Scanf函数有一个\\n,其作用是什么?
scanf("%s",str);\/*此处输入:I love you! *\/ printf("%s",str); Sleep(5000);\/*这里等待5秒,告诉你程序运行到什么地方*\/ \/*不是sleep(5) 1,函数名是Sleep不是sleep。2,C\/C++中,unsigned Sleep(unsigned)应该是毫秒ms. scanf("%s",str1);\/*这两句无需你再输入,是对键盘盘缓冲区再扫描 *\/ sca...

C语言的scanf("%d\\n",&r);只要加入\\n逻辑就错了,有大神能告诉我为什么...
\\n就结束输入了,scanf函数的第一个参数是都要输入的,你这个"%d\\n",就需要先输入一个整数,再输入一个"\\",再输入一个"n",再回车,所以一直阻塞在这,随便搜的一个博客,看看应该就理解了 网页链接

C语言中scanf("%d\\n")中的\\n存在与不存在对于编译结果不同的详细解释...
scanf("%d\\n",&i);printf("%d\\n",i);(1)如果 你 拍入 1 和 Enter 键,scanf("%d\\n",&i); 会等待,再拍 Enter 键 Enter 键 Enter 键 ... 它还在 等待。(2)如果 你 拍入 1 空白 2 和 Enter 键,2个句语句 便执行完毕,打印出 1 多余拍入 的 2 留在 输入缓冲...

为什么输入的时候scanf("%c",&st[i]); %c之前不加\\n就不好使
这可能是键盘缓冲区的问题。请在scanf前面加入 fflush(stdin);您再试试~~

相似回答