C语言-getchar()、getch()、scanf()对比(附C语言代码)

C语言getchar()、getch()、scanf()对比(附C语言代码)

由于最近收到刚入门c语言的小朋友关于 “使用scanf()语句接收char类型数据时出现了一系列迷惑输出” 的提问, 故重新整理了一下C语言中关于getchar()getch()scanf()三个函数的相关知识点.

[toc]

下面从函数的生命周期理解三个函数


getchar()

当函数开始运行的时候,去stdio缓冲区看是否有字符,有则获取一个字符,返回给程序,函数结束。

如果stdio缓冲区为空,则等待从键盘输入数据,同时每输入一个字符都会回显到屏幕上,只有当按下回车键(‘\n’),才将按下回车键之前包括回车键的数据存到stdio缓冲区, 然后getchar()stdio缓冲区获取第一个字符,并返回给程序,同时缓冲区的字符指针往下移动一位(即指针指到已经获取字符的下一个字符),函数结束。(说明:getchar(),可以获取任意字符,包括换行键【ASCII码:10】,空格键【ASCII码:32】等)

代码

#include<stdio.h>
int main(void)
{
    int a,b,c;
    printf("please input num a:\n");
    a=getchar();
    printf("please input num b:\n");
    b=getchar();
    printf("please input num c:\n");
    c=getchar();    
    printf("%d,%d,%d",a,b,c);
}

运行结果

please input num a:
1 
please input num b:
please input num c:
2
49,10,50

运行结果解释:

程序运行打印完“please input num a:”后,进入getchar()函数,由于此时stdio缓冲区没有字符,所以程序等待用户从键盘输入;

当输入“1”并按回车键后,“1”回车键ASCII码存进stdio缓冲区,然后程序第6行的getchar()函数获取“1”存入变量a中,函数运行结束,继续往下运行.

打印出第7行的输出内容,进入第8行的getchar()函数,由于此时,stdio缓冲区还有一个刚才输入的回车键getchar()直接把回车键的内容取走存入变量b中.

继续运行程序第9行的打印内容,然后进入第10行的getchar()函数,此时stdio缓冲区为空,所以需要等待用户从键盘输入,这里输入“2”,按下回车键getchar()函数从stdio缓冲区获取“2"存入变量c中.

最后打印结果为”49“”10“”50“,分别表示字符”1“”回车键“”2“ASCII码。
要想达到想要的结果可以这样修改程序,在每次获取字符后多加一个getchar(),获取stdio缓冲区中每次输入完字符附加的”回车键“。
改进代码

#include"stdio.h"
int main(void)
{
    int a,b,c;
    printf("please input num a:\n");
    a=getchar();
    getchar();
    printf("please input num b:\n");
    b=getchar();
    getchar();
    printf("please input num c:\n");
    c=getchar();
    getchar();
    printf("%d,%d,%d\n",a,b,c);
}

运行结果

please input num a:
1
please input num b:
2
please input num c:
3
49,50,51

也可以在每次getchar()之后加入fflush(stdin)清空stdio缓冲区,也可以达到同样的效果。

改进代码

#include"stdio.h"
int main(void)
{
    int a,b,c;
    printf("please input num a:\n");
    a=getchar();
    fflush(stdin);
    printf("please input num b:\n");
    b=getchar();
    fflush(stdin);
    printf("please input num c:\n");
    c=getchar();
    fflush(stdin);
    printf("%d,%d,%d\n",a,b,c);
}

getch()

getch()这个函数的直接等待获取从键盘输入的字符,并且不回显在屏幕,并且不用以回车键结尾,没输入完一个按键,程序获取该按键的ASCII码并返回,该函数就结束。

代码

#include "stdio.h"
#include  <conio.h>
int main(void)
{
    int a,b,c;
    printf("please input num a:\n");
    a=getch();  
    printf("please input num b:\n");
    b=getch();
    printf("please input num c:\n");
    c=getch();
    printf("%d,%d,%d\n",a,b,c);
}

运行结果

please input num a:
please input num b:
please input num c:
49,32,50

运行结果解释:

依次输入”1“”空格键“”2“结果如上,每次输入的字符不回显在屏幕,不用等待回车键即可获取键盘输入的字符,而是将输入字符的ASCII码打印出来。


scanf()

scanf()函数和getchar()类似,首先从stdio缓冲区读取字符,如果缓冲区有符合格式要求的数据,则获取数据,返回给程序,函数结束;

如果缓冲区没有符合要求的数据(这里不一定为空,因为有时候缓冲区不为空,但是没有否和格式的数据,例如:缓冲区存在空格或者回车,但是返回类型为整型参数),则等待从键盘输入数据,并且必须以回车键结束,输入的数据存入stdio缓冲区,(至少输入一个满足格式要求的数据才可以以回车键结束输入,否则按回车键,继续等待输入)然后scnaf()从缓冲区获取一个数据并返回,函数结束。

代码

#include "stdio.h"
int main(void)
{
    int a,b,c;
    printf("please input num a:\n");
    scanf("%d",&a);
    printf("please input num b:\n");    
    scanf("%d",&b);
    printf("please input num c:\n");
    scanf("%d",&c); 
    printf("%d,%d,%d\n",a,b,c);
}

运行结果

please input num a:

    1  2
please input num b:
please input num c:

        3
1,2,3

运行结果解释:

首先显示程序第5行的打印内容,然后程序执行第6行的scanf()函数,由于此时stdio缓冲区为空,所以等待用户从键盘输入;

这里首先输入了一个回车键发现函数并没有结束,继续等待输入,这就说明,在回车键结束输入之前必须至少输入一个满足scanf()格式要求的数据;

接着输入了几个空格之后输入”1“,再输入两个空格按键之后输入”2“,然后按回车键,就可以结束输入;

然后第6行的scanf()函数获取”1“存入变量a中;

接着程序执行第7行的打印内容,发现第8行似乎没有执行,直接打印了第9行的内容,这里其实第8行的scanf()已经执行,因为此时stdio缓冲区中还有一个输入的数据”2“,第8行的scanf()直接从缓冲区读取”2“存入变量b中,而不需要等待用户从键盘输入;

然后程序就执行了第9行的打印内容,接着执行第10行的scanf()函数,此时,stdio没有满足格式的数据,所以需要等待用户从键盘输入,这里先输入了回车键和几个空格按键,然后输入”3“,再输入回车键. scanf()获取”3“,并存入变量c;

最后打印变量abc的结果,整型形式:123

这里需要指出scanf()函数在从stdio缓冲区中取数据的时候,会清除在找到否和格式要求的数据之前的所有字符,其实就是将指针一直移到获取数据的下一位,测试程序如下

代码

#include "stdio.h"
int main(void)
{
    int a,b,c;
    printf("please input num a:\n");
    scanf("%d",&a);
    printf("getchar():%d\n",getchar());
    printf("please input num b:\n");    
    scanf("%d",&b);
    printf("please input num c:\n");    
    scanf("%d",&c); 
    printf("%d,%d,%d",a,b,c);
}

运行结果

please input num a:

    1  2
getchar():32
please input num b:
please input num c:

        3
1,2,3

运行结果解释:

在前两个scanf()之间加入了getchar(),获取的值是32,即空格ASCII码,所以它获取的是”1“”2“之间输入的空格,而”1“之前的回车键没有获取到,说明已经被第一个scanf()函数从stdio缓冲区取走,或者理解为指针已经指向”1“之后的地址.

关于scanf()获取字符

测试代码

#include "stdio.h"
int main(void)
{
    char a,b,c;
    printf("please input num a:\n");
    scanf("%c",&a);
    printf("please input num b:\n");
    scanf("%c",&b);
    printf("please input num c:\n");
    scanf("%c",&c);
    printf("%c,%c,%c\n",a,b,c);
}

运行结果

please input num a:
a
please input num b:
please input num c:
b
a,
,b

运行结果解释:

输入a之后按回车键,直接运行第7行和第9行的打印内容,似乎第8行没有运行;

这里其实和getchar()函数一样,在输入a之后按回车键a回车键一起进入stdio缓冲区;

第6行scanf()函数从缓冲区获取字符a存入a变量,然后执行第7行,第8行执行的时候,因为此时缓冲区有一个回车键scanf()函数获取回车键存入变量b,所以这里不需要等待用户输入直接进入第9行打印;

之后第10行程序发现缓冲区没有数据,就需要用户输入,这里输入”b“存入变量c,然后依次打印变量abc, 可以看出: 先输出”a“,然后换行,再输出”b“.

关于改进,同样可以使用getchar(),获取输入完数据必须输入的回车键;或者使用fflush(stdin)清空缓冲区.

小朋友的问题

代码

#include <stdio.h>
// 1,2
// 1.2,3.4
// 666666,888888
// 123,456
// A,a
int main(void)
{
   int nA,nB;
   float fC,fD;
   long lE,lF;
   unsigned int u,v;
   char ch1,ch2,ch3,ch4,ch5,ch6;
   scanf("%d,%d",&nA,&nB);
   scanf("%f,%f",&fC,&fD);
   scanf("%ld,%ld",&lE,&lF);
   scanf("%o,%o",&u,&v);
   scanf("%c,%c,%c,%c,%c,%c",&ch1,&ch2,&ch3,&ch4,&ch5,&ch6);
   printf("\n");
   printf("a=%7d,b=%7d\n",nA,nB);
   printf("c=%10.2f,d=%10.2f\n",fC,fD);
   printf("e=%17ld,f=%17ld\n",lE,lF);
   printf("u=%o,v=%o\n",u,v);
   printf("c1=%c,c2=%c,c3=%c,c4=%c,c5=%c,c6=%c\n",ch1,ch2,ch3,ch4,ch5,ch6);
   return 0;
}

运行结果

1,2
1.2,3.4
666666,888888
123,456
A,a,B,b,C,c

a=      1,b=      2
c=      1.20,d=      3.40
e=           666666,f=           888888
u=123,v=456
c1=
,c2=,c3=,c4=,c5=,c6=

问题:

为什么程序第24行的关于ch1,ch2,ch3,ch4,ch5,ch6的输出不符合预期?

问题分析:

程序在执行第17行代码时, 用户在输入123,456之后按回车键,此时123,456回车键一起进入stdio缓冲区;第17行scanf()函数从缓冲区获取字符123,456分别存入u,v变量,然后执行第18行代码的时候,因为此时缓冲区有一个回车键scanf()函数获取回车键存入变量ch1,同时因为读取到回车键, 导致结束次行代码开始执行后续代码, 所以这里仅获取回车键存入字符型变量ch1,而ch2,ch3,ch4,ch5,ch6未存入任何值.(即为空值)

改进建议:
可以使用getchar(),获取输入完数据必须输入的回车键.

改进后代码

#include <stdio.h>
int main(void)
{
   int nA,nB;
   float fC,fD;
   long lE,lF;
   unsigned int u,v;
   char ch1,ch2,ch3,ch4,ch5,ch6;
   scanf("%d,%d",&nA,&nB);
   scanf("%f,%f",&fC,&fD);
   scanf("%ld,%ld",&lE,&lF);
   scanf("%o,%o",&u,&v);
   getchar();
   scanf("%c,%c,%c,%c,%c,%c",&ch1,&ch2,&ch3,&ch4,&ch5,&ch6);
   printf("\n");
   printf("a=%7d,b=%7d\n",nA,nB);
   printf("c=%10.2f,d=%10.2f\n",fC,fD);
   printf("e=%17ld,f=%17ld\n",lE,lF);
   printf("u=%o,v=%o\n",u,v);
   printf("c1=%c,c2=%c,c3=%c,c4=%c,c5=%c,c6=%c\n",ch1,ch2,ch3,ch4,ch5,ch6);
   return 0;
}

运行结果

1,2
1.2,3.4
666666,888888
123,456
A,a,B,b,C,c

a=      1,b=      2
c=      1.20,d=      3.40
e=           666666,f=           888888
u=123,v=456
c1=A,c2=a,c3=B,c4=b,c5=C,c6=c

由于本人水平有限,如有理解或描述错误,还请各位批评指正.
邮箱: 517093978@qq.com

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,723评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,003评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,512评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,825评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,874评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,841评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,812评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,582评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,033评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,309评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,450评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,158评论 5 341
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,789评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,409评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,609评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,440评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,357评论 2 352

推荐阅读更多精彩内容