您的位置:首页 > 其它

C 中scanf ( ) 函数用法 用法

2015-07-15 11:01 295 查看
我觉得,在输入输出函数中,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!

输出: i

10

love!

//因为 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!

输出:

I

love

c!

那么,怎么来输入一个有空格的字符串?

用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下.
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: