C gets()、scanf和getchar()的区别

程序一:(gets())

# include<stdio.h>
int main()
{
       char str1[20],str2[20];
       gets(str1);
       printf("%s\n",str1);
       gets(str2);
       printf("%s\n",str2);
       return 0;
}

测试:

 Hello world! [输入]

 Hello world! [输出] 

12345 [输入] 

12345 [输出]  

【分析】显然与上一个程序的执行情况不同,这次程序执行了两次从键盘的读入,而且第一个字符串取了Hello world! 接受了空格符,而没有像上一个程序那样分成了两个字符串!所以如果要读入一个带空格符的字符串时因该用gets(), 而不宜用scanf()!

gets()可以接受带空格的字符串!

程序二(scanf)

#include <stdio.h>   
int main()  
{  
char str1[20], str2[20]; 
scanf("%s",str1);  
printf("%s\n",str1);    
scanf("%s",str2);  
printf("%s\n",str2);
return 0;    
 }

程序的功能是读入一个字符串输出,在读入一个字符串输出。可我们会发现输入的字符

串中不能出现空格,例如:

测试一输入:

Hello world! 

输出:

Hello 

world! 

【分析】到此程序执行完毕,不会执行第二次的读取操作!这个问题的原因跟问题一类似,第一次输入Hello world!后,字符串Hello world!都会被读到输入缓冲区中,而scanf()函数取数据是遇到回车、空格、TAB就会停止,也就是第一个scanf()会取出"Hello",而"world!"还在缓冲区中,这样第二个scanf会直接取出这些数据,而不会等待从终端输入。

测试二:

Hello[Enter] 

Hello[输出

world[Enter] 

world[输出

【分析】程序执行了两次从键盘读入字符串,说明第一次输入结束时的回车符被丢弃!即:scanf()读取字符串会舍弃最后的回车符!程序的功能是读入一个字符串输出,在读入一个字符串输出。可我们会发现输入的字符串中不能出现空格,例如: 

测试一
输入: Hello world! 
输出: Hello
           world!  
【分析】到此程序执行完毕,不会执行第二次的读取操作!这个问题的原因跟问题一类似,第一次输入Hello world!后,字符串Hello world!都会被读到输入缓冲区中,而 scanf()函数取数据是遇到回车、空格、TAB就会停止,也就是第一个scanf()会取出"Hello",而"world!"还在缓冲区中,这样第二个scanf会直接取出这些数据,而不会等待从终端输入。
 测试二:
          Hello[Enter]
          Hello[输出] 
          world[Enter] 
          world[输出]  
【分析】程序执行了两次从键盘读入字符串,说明第一次输入结束时的回车符被丢弃!即:scanf()读取字符串会舍弃最后的回车符!
scanf()可以接受不带空格的字符串!

程序三:(getchar())

#include <stdio.h>   
int main()  
{  
char str1, str2; 
str1=getchar();  
        printf("%c\n",str1);   
str2=getchar();  
printf("%d\n",str2);
return 0;    
 }

测试:

输入a

输出a

      10

程序的本意很简单,就是从键盘读入两个字符,然后打印出这两个字符的ASCII码值。可是执行程序后会发现除了问题:当从键盘输入一个字符后,就打印出了结果,根本就没有输入第二个字符程序就结束了。例如用户输入字符'a', 打印结果是9710

程序的本意很简单,就是从键盘读入两个字符,然后打印出这两个字符的ASCII码值。可是执行程序后会发现除了问题:当从键盘输入一个字符后,就打印出了结果,根本就没有输入第二个字符程序就结束了。例如用户输入字符'a', 打印结果是a,10。
【分析】:     
 首先我们呢看一下输入操作的原理, 程序的输入都建有一个缓冲区,即输入缓冲区。一次输入过程是这样的,当一次键盘输入结束时会将输入的数据存入输入缓冲区,而cin函数直接从输入缓冲区中取数据。正因为cin函数是直接从缓冲区取数据的,所以有时候当缓冲区中有残留数据时,cin函数会直接取得这些残留数据而不会请求键盘输入,这就是例子中为什么会出现输入语句失效的原因!     
 其实这里的10恰好是回车符!这是因为scanf()和getchar()函数是从输入流缓冲区中读取值的,而并非从键盘(也就是终端)缓冲区读取。而读取时遇到回车(/n)而结束的,这个/n会一起读入输入流缓冲区的,所以第一次接受输入时取走字符后会留下字符/n,这样第二次的读入函数直接从缓冲区中把/n取走了,显然读取成功了,所以不会再从终端读取!这就是为什么这个程序只执行了一次输入操作就结束的原因! 
getchar()可以接受单个字符!


总结:

第一:

要注意不同的函数是否接受空格符、是否舍弃最后的回车符的问题

读取字符时:

scanf()SpaceEnterTab结束一次输入,不会舍弃最后的回车符(即回车符会残留在缓冲区中)

getchar()Enter结束输入,也不会舍弃最后的回车符; 

第二:

为了避免出现上述问题,必须要清空缓冲区的残留数据,可以用以下的方法解决:

方法1C语言里提供了函数清空缓冲区,只要在读数据之前先清空缓冲区就没问题了!

这个函数是fflush(stdin)

方法2:自己取出缓冲区里的残留数据。要注意不同的函数是否接受空格符、是否舍弃最后的回车符的问题! 读取字符时: scanf()以Space、Enter、Tab结束一次输入,不会舍弃最后的回车符(即回车符会残留在缓冲区中); getchar()以Enter结束输入,也不会舍弃最后的回车符;  



  • 1
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值