| 问题描述一:(分析scanf()和gets()读取字符) |
----------------------------------------------------
scanf(), getchar()等都是标准输入函数,一般人都会觉得这几个函数非常简单,没什么特殊的。但是有时候却就是因为使用这些函数出了问题,却找不出其中的原因。下面先看一个很简单的程序:
程序1:
#include <stdio.h> int main() { char ch1, ch2; scanf("%c", &ch1); scanf("%c", &ch2); printf("%d %d/n", ch1, ch2); return 0; } |
#include <stdio.h> int main() { char ch1, ch2; ch1 = getchar(); ch2 = getchar(); printf("%d %d/n", ch1, ch2); return 0; } |
在微软的系统中,C程序处理文本是自动转换这两个符号的:
2、写的时候写一个 /n 自动转换为 /r/n
造成一种误解,好象 /n 就是回车加换行
| 问题描述二:(分析scanf()和gets()读取字符串) |
----------------------------------------------------首先我们看一下scanf()读取字符串的问题:
程序2:
#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[Enter]
Hello[输出]
world[Enter]
world[输出]
我们再看一下gets()读取字符串的情况:
用scanf来读取一个字符串时,字符串中是不可以出现空格的,一旦出现空格,后面的数据就会被舍弃从而残留在缓冲区中。其实有另外一个函数是可以接受空格的,那就是gets(),下面我们看一下这个函数的应用,我们把程序2改动一下:
程序3:
#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 [输出]
--------------------------------------------------------
| 问题描述三:(getchar()暂停程序,查看程序执行结果)|
--------------------------------------------------------
不知道大家有没有遇到过这样的问题,有的编译器程序执行完后的结果界面不会停下而是一闪就没了,以至于看不到执行结果。所以很多人在程序最后加上getchar()语句,目的是想让程序执行完后停下来,等待从终端接收一个字符再结束程序。可是发现有时候这样根本没用,程序照样跳出去了。这是为什么呢?
| 【总结】 |
------------------
第一:要注意不同的函数是否接受空格符、是否舍弃最后的回车符的问题!
读取字符时:
scanf()以Space、Enter、Tab结束一次输入,不会舍弃最后的回车符(即回车符会残留在缓冲区中);
getchar()以Enter结束输入,也不会舍弃最后的回车符;
读取字符串时:
scanf()以Space、Enter、Tab结束一次输入
gets()以Enter结束输入(空格不结束),接受空格,会舍弃最后的回车符!
第二:为了避免出现上述问题,必须要清空缓冲区的残留数据,可以用以下的方法解决:
方法1:C语言里提供了函数清空缓冲区,只要在读数据之前先清空缓冲区就没问题了!
这个函数是fflush(stdin)。
方法2:自己取出缓冲区里的残留数据。
(说实话这个语句我也没看懂,呵呵!为什么格式控制是这样的!希望高手指点一下!)
scanf("%[^/n]",string);//直至遇到回车结束; scanf("%[^/t^/n]",string);//直至遇到tab或回车结束;
再转一篇~~~
scanf函数的一般形式:
scanf(格式控制,地址表列)
int scanf(char *format,argument,...);
“格式控制”的含义同printf函数;“地址表列”是由若干个地址组成的表列,可以是变量的地址,或字符串首地址。
scanf()函数返回成功赋值的数据项数,出错时则返回EOF。
格式字符说明:
%a,%A 读入一个浮点值(仅C99有效)
%c 读入一个字符
%d 读入十进制整数
%i 读入十进制,八进制,十六进制整数
%o 读入八进制整数
%x,%X 读入十六进制整数
%c 读入一个字符
%s 读入一个字符串,遇空格、制表符或换行符结束。
%f,%F,%e,%E,%g,%G 用来输入实数,可以用小数形式或指数形式输入。
%p 读入一个指针
%u 读入一个无符号十进制整数
%n 至此已读入值的等价字符数
%[] 扫描字符集合
%% 读%符号
附加格式说明字符表修饰符说明:
L/l 长度修饰符 输入"长"数据
h 长度修饰符 输入"短"数据
W 整型常数 指定输入数据所占宽度
* 表示本输入项在读入后不赋值给相应的变量
scanf的返回值:
scanf的返回值有后面的参数决定,scanf("%d%d", &a, &b);
1、如果a和b都被成功读入,那么scanf的返回值就是2
2、如果只有a被成功读入,返回值为1
3、如果a和b都未被成功读入,返回值为0
4、如果遇到错误或遇到end of file,返回值为EOF。
使用scanf函数时应该注意的问题
1、sacnf()中的变量必须使用地址。
2、scanf()的格式控制串可以使用其它非空白字符,但在输入时必须输入这些字符。
3、在用"%c"输入时,空格和“转义字符”均作为有效字符。
问题一:scanf()函数不能正确接受有空格的字符串?如: I love you!
#include <stdio.h> int main() { char str[80]; scanf("%s",str); printf("%s",str); return 0; } |
输入:I love you!
输出:scanf()函数接收输入数据时,遇以下情况结束一个数据的输入:(不是结束该scanf函数,scanf函数仅在每一个数据域均有数据,并按回车后结束)。
① 遇空格、“回车”、“跳格”键。
② 遇宽度结束。
③ 遇非法输入。
所以,上述程序并不能达到预期目的, scanf()扫描到"I"后面的空格就认为对str的赋值结束,并忽略后面的"love you!".这里要注意是"love you!"还在键盘缓冲区(关于这个问题,网上我所见的说法都是如此,但是,我经过调试发现,其实这时缓冲区字符串首尾指针已经相等了,也就是说缓冲区清空了,scanf()输入一个单词没问题,输入两个单词无论加入多少个getchar()都会弹出?
#include <stdio.h>
int main(void)
{ char name[40];
printf("what's your name?");
scanf("%s", name); printf("%s,hello boy", name); getchar(); getchar(); return 0; } |
为了避免这种情况,可多设几个字符数组分段存放含空格的串。