在分享之前,先来看一道我最近做的一道编程题吧:
一、
1.题目描述
门上安装了带有数字密码盘电子门。已知此门的密码,是一个由字母组成的字符串,可是密码盘上只能输入数字,后又得知,有一套转换规则,将这个字符串中的大写字母提取出来,按照密码字典中字母和数字的对照表,将这些大写字母对应密码字典中的数字求和,将求和的结果作为密码。这套密码转换不算复杂吧?请你编写一个程序帮助他得到的字母串转换成能够打开大门的密码吧!
2.输入
输入第一行只有一个正整数N ,表示密码字符串中可能出现的不同大写字母的个数,其中(1≤N≤26)。
之后的 2N 行:表示密码字典,每两行代表一个大写字母的信息:其中第一行为一个大写字母,第二行为这个大写字母对应的正整数(不大于1000 )。
最后一行:是密码(一个只含有字母的字符串,其长度不超过 30 )。
3.输出
输出只含有一个自然数表示开门的密码(小于 30000 )。
4.样例输入1
3
A
20
C
10
E
111
abAcEmnCAA
5.样例输出1
181
6.运行代码:
#include <stdio.h>
#include <string.h>
struct translate{
char letter;
int num; //num<=1000
}trans[27];
int main(){
int N; //1<=N<=26
int right = 0; //right<30000
char key[31];
int j = 0;
int i;
scanf("%d", &N);
for(i=0; i<N; i++){
scanf("%c", &trans[i].letter);
scanf("%d", &trans[i].num);
}
scanf("%s", key);
for(i=0; i<strlen(key); i++){
if((key[i] >= 'A')&&(key[i] <= 'Z')){
for(j = 0; j < N; j++){
if(trans[j].letter == key[i]){
right += trans[j].num;
}
}
}
}
printf("%d",right);
return 0;
}
看起来好像没什么大问题吧,而且还能运行呢。但当我运行时发现,实际输入时,并不是按我设想的那样进行输入,而是出现了下面的情况:
这又是为什么呢,我细思苦想了好久都没有思路,于是我向我的室友大神发出了求救,他给我提供了一种办法,如下:
#include <stdio.h>
#include <string.h>
struct translate{
char letter;
int num; //num<=1000
}trans[27];
int main(){
int N; //1<=N<=26
int right = 0; //right<30000
char key[31];
int j = 0;
int i;
scanf("%d", &N);
getchar();
for(i=0; i<N; i++){
scanf("%c", &trans[i].letter);
getchar();
scanf("%d", &trans[i].num);
getchar();
}
scanf("%s", key);
for(i=0; i<strlen(key); i++){
if((key[i] >= 'A')&&(key[i] <= 'Z')){
for(j = 0; j < N; j++){
if(trans[j].letter == key[i]){
right += trans[j].num;
}
}
}
}
printf("%d",right);
return 0;
}
如你所见,我只是把每个scanf()函数的调用下加上了getchar(),就愉快的解决了问题。这这这,这又是为什么呢???于是我去查了scanf()的相关应用及介绍。
发现有人是这么描述的:
当你键盘输入的时候,IO 中断会处理你的输入,键盘驱动程序,会在键盘的内存缓冲中存下你的按键,所以当你用Scanf后,第一个输入的数被传入到scanf的第2个参数的内存
中,此时键盘缓冲中还有一个回车,因此当你如果用getchar后,得到的字符是回车。要把这个回车过滤掉,需调用一次getchar把键盘缓冲清空。scanf()函数接受数据时,通常是以回车键作为结束一个数据的输入(空格键,tab键也行).
如果你还不理解,那我们来举个例子:
#include <stdio.h>
int main(){
int n1,n2;
scanf("%d",&n1);
n2=getchar();
printf("n1=%d,n2=%d",n1,n2);
}
录入时请输入 12 和 Enter,其中 n1 = 12,n2 = 10即’\n’的十进制数
若录入时输入 12 和 space 和 Enter,其中 n2 = 32。32即空格键的十进制数。
二、
好的,接下来我们再来了解一下这个 getchar( ) 函数
1.getchar()的用法:
(1)语法:int getchar(void);
(2)返回值:getchar函数的返回值是用户输入的第一个字符的ASCII码,如出错返回-1,且将用户输入的字符回显到屏幕.如用户在按回车之前输入了不止一个字符,其他字符会保留在键盘缓存区中,等待后续getchar调用读取.也就是说,后续的getchar调用不会等待用户按键,而直接读取缓冲区中的字符,直到缓冲区中的字符读完为后,才等待用户按键。
(3)作用:从标准输入流只读取一个字符(包括空格、回车、tab),读到回车符(’\n’)时退出,键盘输入的字符都存到缓冲区内,一旦键入回车,getchar就进入缓冲区读取字符,一次只返回第一个字符作为getchar函数的值,如果有循环或足够多的getchar语句,就会依次读出缓冲区内的所有字符直到’\n’.要理解这一点,之所以你输入的一系列字符被依次读出来,是因为循环的作用使得反复利用getchar在缓冲区里读取字符,而不是getchar可以读取多个字符,事实上getchar每次只能读取一个字符.如果需要取消’\n’的影响,可以用getchar()来清除,如:while((c=getchar())!=’\n’),这里getchar();只是取得了’\n’但是并没有赋给任何字符变量,所以不会有影响,相当于清除了这个字符。
2.如果你还没有理解,那我们来看个简单的例子说明一下:
#include <stdio.h>
int main(){
char a,b;
a=getchar();
b=getchar();
putchar(a);
putchar(b);
}
此时我们输入x-回车-y-回车,结果会是xy吗?
不是的 结果是a=x b=’\n’(回车)
当我们开始打x的时候,x在缓冲区,当我们按下回车的时候,第一个getchar()才获取到x这个值 顺利存到a里 但回车既是确定又是字符,回车’\n’它也跟着进了缓冲区,并且和x一起被释放,x到了程序里 回车(’\n’)也到了程序里,并且被第2个 getchar()获取,此时我们输入y,这时缓冲区里有一个字符 ‘y’, 然后我们又按下回车,‘y’和’\n’又一起准备被释放,由于程序里没有第3个getchar(),所以’y’和’\n’还留在缓冲区,但它们两个已经是在准备状态中,如果程序又出现一个getchar(),‘y’不需要你按回车它会直接进入到第3个getchar(),getchar()是依次获取,按先后顺序,不会先获取’\n’,而’\n’正等待着马上进入第4个getchar()。
3.getchar()与getch()的区别:
用getchar()时,在键盘上按一个字符后,要按回车才能读取进去;
用getch()时,要包含头文件<conio.h>,在键盘上按一个字符马上就被读取进去,不用按回车,因此可以作为“按任意键继续”的执行语句。
**4.补充:**putchar()的用法:
(1)输出:putchar函数只能用于单个字符的输出,向终端输出一个字符,且一次只能输出一个字符。
(2)格式:对于变量来说,格式为:putchar(ch);对于常量来说,格式为:putchar(‘ch’),对于转义字符来说,格式为:putchar(’\n’)。