当前位置:文档之家› C语言 中scanf详解

C语言 中scanf详解

C 中scanf ( ) 函数用法心得我觉得,在输入输出函数中,scanf()函数,应该是最麻烦的,有时它给我们的结果很可笑,但是一定是一原因的....首先声明一下,这篇日志不是介绍scanf()中各种格式符用法的文章(没有这个必要,但是大家一定要会用).我尝试了很多种输入,包括一些错误的练习,曾经对scanf()由迷茫转向清醒,又由清醒再次转向迷茫......不知道何时是个尽头,谁让C如此高深呢?在这里贴出来,也是想让自己时而不时能看到,也想知道自己的理解是否有错,错在哪里(所以我就厚着脸皮,放在上面了).注意, 键盘缓冲区与输入有着密切的关系,并且, 类型匹配对输入也极为重要!!下面进入主题:scanf对流的操作遵从类型匹配操作原则,如果类型不匹配,它将不读取输入流。

因此输入流将滞留,如果输入流不空,scanf不会等待用户输入,直接从缓冲区中输入. 但是,scanf() 怎样匹配? stdin又是什么?在网上搜到的关于匹配的非常少,有些细节原因还是找不到.所以,我自作主张的下了点结论:例: scanf("%d,%d",&i,&j); 输入:12 ,13回车但是,j!=13. //注意,12后有一个空格,why?原因:我解释为,在scanf()中,格式字符串中普通字符(不包括空白字符)实行的是严格匹配,因为格式串中%d后面是一个 ',' ,因此输入中数字12后必须为一个','.scanf("1123%s",&str); 输入:1123aaabb 时str为aaabb,但是,输入24aabbdd时, 会出错,因为1123必须进行严格匹配.另外: scanf("%d\n",&i); printf("i=%d",i); 要怎么输入才能输出:i=12 ? 它不是你想像中的那样,有机会尝试一下吧!一些样例:scanf()是一个有返回值的函数,它的返回值是什么?怎么样利用这个特性?scanf()中的匹配原则: 在本文第五点具体说明...scanf()中各种数据格式匹配的开始条件,结束条件 .如: %d ,\n等类型输入结束条件.scanf("%d\n",&i);printf("%d",i); 输入 12回车,并无输出,why? scanf()函数的结束条件: 当各个格式符匹配完毕,且最后有一个回车时,函数结束. scanf("%s",str)连续输入127个就不能继续输入了. //TC中,VC好像是4000多..//说明键盘缓冲区长度为一个字节吗?但是 stdin->bsize(缓冲区大小)事实上为512,这又是为什么?stdin缓冲区中的数据残留 : scanf("%3s",str); c= getchar(); 输入: aaabbccc回车, 此时str="aaa",c='b'; //缓冲区中数据残留!getch()不经过缓冲区,直接接收键盘上输入的字符.//在上例中,加上一个 ch=getch(); 但是getch()并不能读取bbccc中的任何一个,说明getch()与getchar()并不一样,并且它们对Enter读取的值也不同! 一个不常用的格式符: %[] ,如 scanf("%[a-z]",str);输入: abcdefdsaABCDEF 输出:str="abcdefdsa" ;怎么用scanf()来输入一个有空格的字符串?scanf()处理时,一个Enter送到缓冲区中有两个值 : 一个回车(10) ,一个换行(13). 可以用getchar()来接收(但是,在只能接收到\n,即13).在一个scanf()函数之后加个fflush(stdin)可以清除输入数据残留?scanf("%3s",str); fflush(stdin); c=getchar();直接输入aaabbbddd回车, c还能取得值吗?下面是详细解释:scanf()函数执行成功时的返回值是成功读取的变量数 , 也就是说,你这个scanf()函数有几个变量,如果scanf()函数全部正常读取,它就返回几。

但这里还要注意另一个问题,如果输入了非法数据,键盘缓冲区就可能还个有残余信息问题。

scanf()- 函数是通用终端格式化输入函数,它从标准输入设备(键盘) 读取输入的信息。

可以读入任何固有类型的数据并自动把数值变换成适当的机内格式。

sscanf() - 从一个字符串中读进与指定格式相符的数据.函数原型:Int sscanf( string str, string fmt, mixed var1, mixed var2 ... );int scanf( const char *format [,argument]... );sscanf与scanf类似,都是用于输入的,只是后者以屏幕(stdin)为输入源,前者以固定字符串为输入源。

其中的format可以是一个或多个 {%[*] [width] [{h | l | I64 | L}]type | ' ' | '\t' | '\n' | 非%符号}如: sscanf("123456", "%s", buf);puts(buf); //结果为: 123456下面主要说一下scanf()的用法:scanf函数的一般形式scanf(格式控制,地址表列)int scanf(char *format[,argument,...]);“格式控制”的含义同printf函数;“地址表列”是由若干个地址组成的表列,可以是变量的地址,或字符串首地址。

scanf()函数返回成功赋值的数据项数,出错时则返回EOF。

注: scanf()中空白字符(包括\n,space)会使scanf()函数在读操作中略去输入中的零个或者一个或者多个空白字符,空白符可以是space,tab,换行等等,直到第一个非空白符出现为止。

//下一个格式符为%c也同样如此.如,scanf("%d %c",&i,&ch); 输入:11 A回车,i=11,ch=A. 这里ch 并不为空格.一个非空白字符会使scanf()函数在读入时剔除掉与这个非空白字符相同的字符。

如: scanf(" %c",&ch); 输入: 若干个回车后,输入 A, ch=A.scanf("5729%s",str); 输入: 5729okok str=okok. 但是请注意:当输入的前几个不是5729时(就算以空格开始也不行!),将会出错,str值不变.scanf()中格式字符说明%p 读入一个指针%[] 扫描字符集合%n 至此已读入值的等价字符数%s 读入一个字符串,遇空格、制表符或换行符结束。

%c %d %i %o %x %X %c%f 输入实数,可以用小数形式或指数形式输入。

%F %e %E %g %G%u 读入一个无符号十进制整数%% 读%符号附加格式说明字符表修饰符说明L/l 长度修饰符输入"长"数据h 长度修饰符输入"短"数据W 整型常数指定输入数据所占宽度m指定输入数据所占的宽度* 星号空读一个数据结合实际例程,一一阐述:一: "%d%d%d"是按十进值格式输入三个数值。

输入时,在两个数据之间可以用一个或多个空格、tab键、回车键分隔。

"%d,%d,%d"运行时按如下方式输入三个值:3,4,5↙(输入a,b,c的值)或者3,□4,□5↙(输入a,b,c的值)3,□□□4,□5↙(输入a,b,c的值)................都是合法的,但是输入3□,4□□,5↙出错!!!!//因为scanf()的格式串中普通字符实行完全匹配!%c在用"%c"输入时,空格和“转义字符”均作为有效字符。

二: scanf()函数以一个非空白字符(包括空格,跳格,回车)开始一个数据的输入( %c 当然除外!,但是注意,gets()以任意字符为开始! ).scanf()函数接收输入数据时,遇以下情况结束一个数据的输入:(不是结束该scanf函数,scanf函数仅在每一个数据域均有数据,并按回车后结束)。

①遇空格、“回车”、“跳格”键。

②遇宽度结束。

③遇非法输入。

//这里很重要,如果有非法输入,则结束这个数据的输入,但是输入的非法数据还在缓冲区中,你可以用对应的数据类型接收!也可以干脆清除缓冲区. // 例如:i=10;scanf("%s",str);scanf("%d",&i);scanf("%s",str2);printf("%s\n",str);printf("%d\n",i);printf("%s\ n",str2);输入: i love!输出: i10love!//因为love对 %d来说是一个开始输入(scanf()以一个非空白开始),但是因为不合法,所以这个开始也就是结束,i值不变!三 . scanf()函数能不能正确接受有空格的字符串?如: I love c!//事实上它可以!!!char str[80];scanf("%s",str); //输入I love c? //结果: 输出I分析: scanf()扫描到"I"后面的空格就认为对str的赋值结束,并忽略后面的"love c!" . 这里要注意是"love c!"还在键盘缓冲区经过调试发现,其实这时缓冲区字符串首尾指针已经相等了,也就是说缓冲区清空了,scanf()函数应该只是扫描stdin流,这个残存信息是在stdin中.//其实,我曾试过,用scanf("%s",str)连续输入127个字符后,键盘缓冲区就装不下了,也就是说,对输入的不做处理,继续输入,就没有反应了,只有输入回车才有效.来验证一下:#include <stdio.h> int main(){char str[80];char str1[80];char str2[80];scanf("%s",str); /*此处输入:I love you! */printf("%s",str);sleep(3); /*这里等待3秒,告诉你程序运行到什么地方*/scanf("%s",str1); /*这两句无需你再输入,是对键盘盘缓冲区再扫描 */ scanf("%s",str2); /*这两句无需你再输入,是对键盘盘缓冲区再扫描 */ printf("\n%s",str1);printf("\n%s",str2);return 0;}输入:I love c!输出:Ilovec!那么,怎么来输入一个有空格的字符串?用gets()当然可以,但我们同样可以用 scanf(),因为,scanf()还有一个我们不常用的输入格式符: "%[]"特别的:%*[width] [{h | l | I64 | L}]type 表示满足该条件的被过滤掉,不会向目标参数中写入值支持集合操作: //类似于正则表达式 .%[a-z] 表示匹配a到z中任意字符,贪婪性(尽可能多的匹配) // 或: %[a-z1-9] 遇到非 a~z,1~9的则结束.%[aB'] 匹配a、B、'中一员,贪婪性 // 以非 a,b,' 的字符为结束.%[^a] 匹配非a的任意字符,贪婪性 scanf("%[^a]",str);// 输入: ffddssaaff 则提取 ffddss到字符串str中. 即扫描到a就立即作为结束.故,我们可以用 scanf("%[^\n]",str); 来输入一个有空格的字符串. //输入:l love c!回车, 则str中为I love c!那么 scanf("%*[^/]/%[^@]",str); 的作用呢??? //输入一个字符串,截取 /到@之间的字符串...四: 解决键盘缓冲区的污染问题. //即残余信息 ,这个很重要吧.例如:scanf("%c",&c1);scanf("%c",&c2);当输入: a回车b回车输出c1,c2的值:很明显 c2不为b.原因: 将c2用int表示出来,看看scanf()函数赋给C到底是什么,结果是c2=10 .ASCII值为10是什么?换行即\n.我们每击打一下"Enter"键,向键盘缓冲区发去一个“回车”(\r),一个“换行"(\n).在这里 \r被scanf()函数处理掉了(姑且这么认为吧^_^),而\n被scanf()函数“错误”地赋给了c.// 好像用getch()时,击ENTER,接收的是回车,并且,它不从键盘缓冲区经过,即键盘缓冲区内容与其无关!!!!!我做了个小试验:scanf("%3s",str);puts(str);c1=getch();printf("c1 %d\n",c1);c2=getchar();printf("c2 %d\n",c2);// 输入 aaabcdef 输出aaa 输入 A输出: 65,98 看到了吧!!!// %3s只接收 aaa,然后输入一个A,被getch()接收,输出之后,getchar()继续从缓冲区中取出 b ,明白了.解决这类问题最好的办法是: 可以在两个scanf()函数之后加个fflush(stdin); //功能: 清除一个流用法: int fflush(FILE *stream);另外,百度百科上还有另外一个方法: 用getchar()和getch()接收. 但是,通过上面那个实验我们可以看到,getch()并不对缓冲区作处理,并不能处理scanf()的残余信息.//可以试一下:scanf("%c",c1);getch(); //假设用来接收换行.scanf("%c",c2);//输入A回车后 : c1值为 65,c2 为13 ,即换行.//实际上,getch()会等着一个输入.//而把getch()换为getchar()后, 输入 :A回车B回车,输出A B 五: 下面这段程序要输入两个数,程序才结束,而不是预期的一个,why?#include<stdio.h>int main(){int a;printf("input the data\n");scanf("%d\n",&a); //这里多了一个回车符\n,如果用scanf("%d ",&a); printf("%d",a); //也会出现同样问题. return 0;}//输入: 11回车后,没有输出,再输入空格,回车,Tab 中任意多个,都没有输出,当输入非空白字符时如输入abc回车,才有输出,输出为11.//分析其原因(不一定准确,应该可以这么解释吧):scanf()是一个终端格式化输入函数,也就是说按匹配对变量进行赋值!!规则 : 例如对于 " %d\n" :第一个空格可以与输入缓冲区的任意多个空白字符匹配(包括空格,回车,Tab),当遇见第一个非空白字符时,结束其匹配,接着处理%d .%d可以与连续的数字符号进行匹配,当遇到第一个非数字符号时,结束匹配,若与其匹配的数字个数为0,则%d对应的变量值不变.//注意,%d与任意一个非int字符开始匹配失效,就算是 '.'也不例外,如输入 12.30则 .30不会被读取,而是留在缓冲区中.同理,\n也要与一个或者多个Enter,Tab,space匹配,直到遇到第一个非空白字符.同样对于 "%d": 与缓冲区中第一个非空白字符开始进行匹配.但是"%c"是个例外,它与缓冲区中的第一个字符就匹配,不论空白与否,所以,处理输入的字符+Enter时,一定要请注意其中的Enter.所以对上面的例子,输入为: 11a时,回车一次就可以输出11,但是不要忘了,缓冲区中还有 a和\n !!!六:有关stdin, 事实上它就是一个标准输入文件, 为 File * 类型.因此, scanf("%s",str); 也就等价于 fscanf(stdin,"%s",str);但是 scanf()只能用来输入 127以下个字符,也就是说,缓冲区只能装下127个字符+'\',那为什么 stdin->bsize又为 512呢? //在TC下.。

相关主题