羽夏闲谈—— C 的 scanf 的高级用法


前言

??今天看到博友发了个有关scanf的使用的注意事项,就是讨论缓冲区残存数据的问题,用简单的代码示例复述一下:

#define _CRT_SECURE_NO_WARNINGS

#include 
#include 

int main(int argc, char *argv[])
{
    int a;
    char b;
    scanf("%d", &a);
    scanf("%c", &b);
    printf("a = %d , b = %d\n", a, b);
    system("pause");
    return 0;
}

??你或许碰到这个输出:

5
a = 5 , b = 10
请按任意键继续. . .

??我明明想输入个5,然后回车输入下一个字符,但是,回车符也是个字符,会被scanf进去,绝大多数人的解决方案就是提前把这个字符读取走,但是如果缓冲区的东西太多的话,需要加个循环,才能处理,下面我来介绍scanf的高级用法,之间研究过,忘记在哪里看到的,这个是我总结的,那些基础用法自己看看书就行了。

清空缓存区

??我先把清空缓冲区的代码放上,因为后面的代码都会用到,至于为什么后面会有详细介绍:

scanf("%*[^\n]"); //清除到回车符的所有字符
scanf("%*c");  //清除回车符

指定输入长度

??我们都知道scanf可以指定小数位数和长度,如下是代码示例:

#define _CRT_SECURE_NO_WARNINGS

#include 
#include 

int main(int argc, char *argv[])
{
    int n;
    float f;
    char str1[23];
    scanf("%2d", &n);
    //清空缓冲区代码{
    scanf("%*[^\n]");
    scanf("%*c");
    //}
    scanf("%5f", &f);
    //清空缓冲区代码{
    scanf("%*[^\n]");
    scanf("%*c");
    //}
    scanf("%5s", str1);
    puts("执行后:");
    printf("n=%d, f=%g, str=%s\n", n, f, str1);
    
    system("pause");
    return 0;
}

??如下是输入和输出结果:

5653 12
2.56458452 2356.9999
helloworld
执行后:
n=56, f=2.564, str=hello
请按任意键继续. . .

??看到没,清空缓冲区的代码有效果了,如果没有这行清空缓冲区的代码,就会成这样子:

5653 12
执行后:
n=56, f=53, str=12
请按任意键继续. . .

??这就是清除缓冲区的作用。

单范围匹配

??不要惊讶,scanf也是支持类似正则表达式的功能的,我们用如下代码进行演示:

#define _CRT_SECURE_NO_WARNINGS

#include 
#include 

int main(int argc, char *argv[])
{
    char str2[30];
    scanf("%[abcd]", str2); //%[abcd]表示只要字符串只有 a,b,c,d 范围内就匹配
    puts("执行后:");
    printf("%s\n", str2);
    system("pause");
    return 0;
}

??如下是输入和输出结果:

babccbaxyz
执行后:
babccba
请按任意键继续. . .

??代码中的注释可能说的不明白,这里我长篇大论一下:字符串从开头开始匹配,必须字符串只有abcd中这四个字符任意一个才有效,如果开头没有这四个字符,则匹配为空。

多范围匹配

??既然支持单范围了,肯定也支持多范围,什么是多范围匹配可以先看看一些基础示例,如果会正则表达式的话很容易懂:

  • %[a-z]表示读取 abc...xyz 范围内的字符,也即小写字母;
  • %[A-Z]表示读取 ABC...XYZ 范围内的字符,也即大写字母;
  • %[0-9]表示读取 012...789 范围内的字符,也即十进制数字;
  • %[a-zA-Z]表示读取大写字母和小写字母,也即所有英文字母;
  • %[a-z-A-Z0-9]表示读取所有的英文字母和十进制数字;
  • %[0-9a-f]表示读取十六进制数字

??如果经常使用的话也就这些类型,如下是代码示例:

#define _CRT_SECURE_NO_WARNINGS

#include 
#include 

int main(int argc, char *argv[])
{
    char str3[30];
    scanf("%[a-zA-Z]", str3); //只读取字母
    puts("执行后:");
    printf("%s\n", str3);
    system("pause");
    return 0;
}

??如下是输入和输出结果:

abcXYZ123abcXYZ123
执行后:
abcXYZ
请按任意键继续. . .

不匹配某些字符

??既然有匹配的字符,肯定有不想匹配的字符。对于不匹配某些字符,scanf允许我们在%[ ]中直接指定某些不能匹配的字符,具体方法就是在不匹配的字符前面加上^,给几个例子:

  • %[^\n]表示匹配除换行符以外的所有字符,遇到换行符就停止读取;
  • %[^0-9]表示匹配除十进制数字以外的所有字符,遇到十进制数字就停止读取。

??如下是代码示例:

#define _CRT_SECURE_NO_WARNINGS

#include 
#include 

int main(int argc, char *argv[])
{
     char str11[30], str12[30];
    scanf("%[^0-9]", str11);

    scanf("%*[^\n]");
    scanf("%*c");           //清空缓冲区
    
    scanf("%[^\n]", str12); //等效为gets()
    puts("执行后:");
    printf("str1=%s \nstr2=%s\n", str11, str12);
    system("pause");
    return 0;
}

??如下是输入和输出结果:

abcXYZ@#87edf
Cnblog wingsummer
执行后:
str1=abcXYZ@#
str2=Cnblog wingsummer
请按任意键继续. . .

丢弃数据

??scanf还允许把读取到的数据直接丢弃,不往变量中存放,具体方法就是在%后面加一个*,如下是代码示例:

#define _CRT_SECURE_NO_WARNINGS

#include 
#include 

int main(int argc, char *argv[])
{
    int nn;
    char str[30];
    scanf("%*d %d", &nn);
    scanf("%*[a-z]");
    scanf("%[^\n]", str);
    puts("执行后:");
    printf("n=%d, str=%s\n", nn, str);
    system("pause");
    return 0;
}

??如下是输入和输出结果:

100 999abcxyzABCXYZ
执行后:
n=999, str=ABCXYZ
请按任意键继续. . .