scanf的各种用法小结


[转]scanf的用法1

必须先声明一下,是转自他人网页

虽然有人说在大型程序设计中永远不要使用scanf(),
但是对于习惯用C的人和参加ACM/ICPC的人,
scanf()还是用的比较多的,而且用好了用起来也很方便
 
下面整理一点儿平时常需要用到而可能有些人一直理不清的一些知识
 
一般常识:
scanf()函数的一般格式为:scanf("格式字符串",输入项首地址表)
scanf的格式控制的一般形式为:%[*][宽度][F|N][h|l]类型字符
  • []中的控制字符为可选项 
  • "*"表示该输入项读入后不赋予任何变量,即跳过该输入值。这在减小内存开支上面还是有一点用处的,不需要的字符直接跳过,免得申请没用的变量空间 
  • "宽度"表示输入读入字符的长度,对于整型表示截取相应宽度的数字赋给后面列表中的相应变量;对于字符型表示读入相应长度的字符后把第一个字符赋给相应的变量,其余的自动舍弃。例如scanf("%2d%3d",&a, &b);如果输入为12345则将12赋给a,将45赋给b;scanf("%2c%3c",&a, &b);如果输入为12345则将'1'赋给a,将'3'赋给b 
  • 、N、h、l分别表示远指针、近指针、短整和长整型,对于_int64相应的控制字符为ll或I64 
  • "类型字符"为 d -- 输入十进制整数 、o -- 输入八进制整数 、x -- 输入十六进制整数 、u -- 输入无符号十进制整数 f或e -- 输入实型数(用小数形式或指数形式) 、c -- 输入单个字符 、s -- 输入字符串 
字符串读入的一些技巧
对于输入字符串还有一些比较有用的控制,
例如经常需要读入一行字符串,而这串字符里面可能有空格、制表符等空白字符,
如果直接用%s是不可以的,于是有些人就想到用gets(),当然这也是一种选择,
但是懂C的人基本上都知道gets()是一个很危险的函数,而且很难控制,
特别是与scanf()交替使用时前者的劣势更是一览无余,所以gets()一般是不推荐用的,
其实用%[^/n]就可以很好的解决这个问题了,
^表示"非",即读入其后面的字符就结束读入。
这样想读入一行字符串直接用 scanf("%[^/n]%*c",str); 就可以了
%*c的作用是读入/n,否则后面读入的将一直是/n。
所有对%s起作用的控制都可以用%[],
比如%[0-9]表示只读入'0'到'9'之间的字符,%[a-zA-Z]表示只读入字母,
'-'是范围连接符,
当然也可以直接列出你需要读入的字符,上面读字母之所以用范围连接符是因为
要输入52个字符太麻烦了,
如果你只需要读"abc"里面的字符就可以用%[abc] (或者%[cab]、%[acb]、%[a-c]、%[c-a].....),
如果想读入某个范围之外的字符串就在前面加一个'^',如:%[^a-z]就表示读入小写字母之外的字符
上面这些用法其实可以有很多推广用法的,比如说你要处理下面的字符串
23 44r f30
88888,3245;34:123.
让你输出里面所有的数字,就可以用下面的代码:
#include <stdio.h>
bool skip(){
    scanf("%*[^0-9]");
    return true;
}
int main(){
    int n;
    while (skip() && scanf("%d", &n)!=EOF)
        printf("%d/n", n);
    return 0;
}

我曾经在这个函数上犯过不少错误,也看到别人犯过的错误,记下来,提醒自己不要重蹈覆辙了。如果对你有用,那就更好了:)如果你发现文章中有错误,欢迎你不吝赐教。希望和大家一起学习
有关详细的scanf函数解释,大家可以去看看《C程序设计语言》(K&C)和《C语言大全》(后面我把其中scanf的部分贴了出来)。


曾经错的几个地方:(xpsp2vc6.0环境下)
1.
空白符问题
#include<stdio.h>
main()
{   
int a;
printf("input the data\n");
scanf("%d\n",&a);//
这里多了一个回车符\n
    printf("%d",a);
return 0;
}
结果要输入两个数程序才结束,而不是预期的一个。why
原因:用空白符结尾时,scanf会跳过空白符去读下一个字符,所以你必须再输入一个数。这里的空白符包括空格,制表符,换行符,回车符和换页符。所以如果你用scanf("%d   ",&a)也会出现同样的问题。
解决方法:这种错误大多是输入的时候不小心,多注意一点就好了。这种问题也不好检查,编译没有问题,一个空格也不容易看出来。当你的程序出现上面的问题时,自己对照检查一下就可以了。


2.
缓冲区问题
这是一个非常容易错的地方,我就错过多次。
#include <stdio.h>
main()
{
int n = 5;
char c[n];
for(int i = 0; i < n; i++)
   c[i] = scanf("%c",&c[i]);
printf(c);
return 0;
}
如果输入:
a
b
c
那么循环就会提前结束了.
原因:输入a和第一个回车后,a和这个回车符都留在缓冲区中。第一个scanf读取了a,但是输入缓冲区里面还留有一个\n,第二个scanf读取这个\n(ascii码值为10,可用%d打印出来)。然后输入b和第二个回车,同样的,第三个scanf读取了b,第四个scanf读取了第二个回车符。第五个读取了c。所以五个scanf都执行了,并没有提前结束。只不过有的scanf读取到了回车符而已。
解决方法:把程序改成这样就可以了:
for( i = 0; i < n; i++){
   scanf("%c",&c[i]);
fflush(stdin);//
刷新缓冲区       /* 这个用法是在有些编译器下有效,如VC6,有些下无效,如gcc 4.4.7*/

可以这样处理

for(i = 0; i < n; i++){
           scanf("%c",&c[i]);

           scanf("%[^\n]");    /*  表示如果一经发现输入的字符属于方括号内字符串中某个字符,那么就结束提取;如果不属于就提取该字符。该方法会自动加上一个字符串结束符到已经提取的字符后面。  这样可以在scanf提取一个字符号,把包括\n在内的字符。相当于清空输入缓冲区
*/
}

或者使用空格占位,

for(i = 0; i < n; i++){
           scanf(" %c",&c[i]);   /* %前面有一个空格*/
}


或者不用scanf,而用gets()函数,如:
#include<stdio.h>
main()
{   
char c[5];
gets(c);      /* 读取一行,知道EOF或者\n,并替换为\0,  不过编译器(gcc 4.4.7)提示次函数危险,不建议用*/
printf(c);
return 0;
}
但要注意:这个函数自动把你最后敲的回车转换为字符'\0'。如果你的输入超过了数组的大小,那么就会产生错误。


3.scanf()
函数的参数输入类型不匹配问题
这是我在csdn论坛上见到的问题,这个错误有时候会让人莫名其妙。
#include<stdio.h>
main()
{
int a=123;
char c='t';
printf("input\n");
scanf("%d%c",&a,&c);
scanf("%d%c",&a,&c);
scanf("%d%c",&a,&c);
printf("%d\n%c\n",a,c);
return 0;
}
当输入a回车后,会直接跳过下面2个scanf语句,直接输出为
123
t
原因:对于scanf("%d%c",&a,&c)scanf语句执行时,首先试图从缓冲区中读入一个%d类型的数据,如果和第一个参数匹配,则继续从缓冲区中读取数据和第二个参数进行匹配,依次进行下去,直到匹配完所有的参数;如果其中有一个参数不匹配,那就从这个地方跳出,忽略这个scanf后面所有的参数,而去执行下一条语句。
可以用下面的程序验证一下:
#include <stdio.h>
int main()
{
int a=123,b=1;
char c='t';
    scanf("%d%d",&a,&b);
scanf("%c",&c);
printf("%d\n%d\n%c\n",a,b,c);
return 0;
}
输入:2回车a回车
结果是:
2
1
a
解决方法scanf()函数执行成功时的返回值是成功读取的变量数,也就是说,你这个scanf()函数有几个变量,如果scanf()函数全部正常读取,它就返回几。但这里还要注意另一个问题,如果输入了非法数据,键盘缓冲区就可能还个有残余信息问题。
比如:
#include <stdio.h>
main()
{
int a=123,b;
while(scanf("%d%d",&a,&b)!=2)
   fflush(stdin);
printf("%d\n%d\n",a,b);
return 0;
}
你可以试一下,如果输入不是数字时,会有什么反应。

补充scanf中一种很少见但很有用的转换字符:[...][ ^...]
#include<stdio.h>
main()
{
char strings[100];
scanf("%[1234567890]",strings);
printf("%s",strings);
return   0;
}
运行,输入:1234werew后,结果是:1234
通过运行可以发现它的作用是:如果输入的字符属于方括号内字符串中某个字符,那么就提取该字符;如果一经发现不属于就结束提取。该方法会自动加上一个字符串结束符到已经提取的字符后面。

注意:方括号两边不能空格,如:scanf("%[ 1234567890 ]",strings); scanf("%[ ^1234567890 ]",strings);不让空格也会算在里面的。
用这种方法还可以解决scanf的输入中不能有空格的问题。只要用
scanf("%[^\n]",strings);
就可以了。很神奇吧。


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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值