C语言scanf函数奇遇记

看《The C Programming Language》中关于scanf函数部分时随意敲了几行代码,本以为简单的不得了,都有点“不屑于”敲,却没想到这一敲竟然敲出个不小的问题,涉及到好多东西啊,哈哈!下面把我这次的经历和大家分享一下,希望也能对大家有所帮助。

    一、代码实例

    我当时敲的代码:

    点击(此处)折叠或打开

    #include<stdio.h>

    int main()

    {

    int a;

    int b;

    char mon[20];

    int count;

    count = scanf("%d,%s,%d", &a,mon,&b);

    printf("%d,%s,%d\n",a,mon,b);

    printf("%d\n",count);

    return 0;

    }

    运行结果:

    ocean@ocean-desktop:~/桌面$ ./re

    12,fefe,45   /*这是我的输入*/

    12,fefe,45,10359588

    2

    结果看起来挺像我们想要的结果的,只是最后多了个奇怪的数字;但仔细看下count的值我们就纳闷了,怎么是2不是3呢?怎么scanf只读了两个值?到底怎么回事呢?先用gdb调试一下吧,看看a和mon里都是些什么。

    二、GDB调试情况

    (gdb) p a

    $1 = 12

    (gdb) p mon

    $2 = "fe,45\000\377\277\245\324\025\000\060\340\021\000K\205\004\b"

    明白了吧?原来fe,45作为一个整体被存到mon里了,b根本没读到值,显示了个原内存里的乱七八糟的数值(不相信的话可以在程序开头给b赋个值,最后结果肯定是输出当初赋的值,因为根本没有给b读入新的值),scanf真的只读了两个值,所以count显示2。那为什么会这样呢?让我们来看看scanf函数的相关信息吧。

    三、scanf函数工作原理

    scanf()是从输入流缓冲区中读取值的,而并非从键盘(也就是终端)缓冲区读取。往输入流缓冲区送数据是遇到回车(\n)而结束的,这个\n会一起读入输入流缓冲区。scanf() 开始读取输入以后,会在遇到的第一个空白字符空格(blank)、制表符(tab)或者换行符(newline)处停止读取。

    格式控制字符串中有普通字符(非格式字符)时,这些字符作为输入数据的分隔符,在scanf函数读入数据时自动去掉。

    scanf()格式控制字符串中如果使用%s说明符,那么空白字符以外的所有字符都是可以接受的,所以scanf() 跳过空白字符直到遇到第一个非空白字符,然后保存再次遇到空白字符之前的所有非空白字符。这就意味着%s使scanf() 读取一个单词,也就是说,一个不包含空白字符的字符串。

    好,让我们分析下上述的结果是如何出现的吧。

    四、原因分析

    首先,scanf()跳过空白字符(这里没有,因为第一个字符就是1)直到遇到一个非空白字符1,然后继续读2,读到逗号这个非数字符号时scanf知道整数读完了,将12赋给a,此时输入流缓冲区中第一个开头的字符是逗号;scanf继续读,读到逗号与格式控制字符串的逗号匹配,pass;从f继续读,一直读到下一个空白符——我们结束时敲的回车(scanf自动把这个回车符去掉了,没有送到字符串里),字符串读完了,此时输入流缓冲区里第一个开头的字符是我们敲的回车符;继续读,回车符与格式控制字符串里的逗号不批配,读取失败,不读了。   综上所述,scanf确实只读了一个整数和一个字符串,返回值是2。

    那有什么办法实现用逗号作为间隔符的情况呢?下面提供两种方法:

    五、解决方法

    法1:

    scanf("%d,%[^,],%d", &a,mon,&b);

    printf("%d,%s,%d\n",a,mon,b);

    相关知识:scanf中一种很少见但很有用的转换字符:[…]和[ ^…]

    %[…]如果输入的字符属于方括号内字 符串中某个字符,那么就提取该字符;如果一经发现不属于就结束提取。%[^…]如果一经发现输入的字符属于方括号内字符串中某个字符,那么就结束提取;如果不属于就提取该字符。这两种方法会自动加上一个字符串结束符到已经提取的字符后面。例如:

    点击(此处)折叠或打开

    #include<stdio.h>

    main()

    {

    char strings[100];

    scanf("%[1234567890]",strings);

    printf("%s",strings);

    return 0;

    }

    运行,输入:1234werew后,结果是:1234。

    采用这种方法,读完fefe后遇到逗号便结束字符串的读取,继续读时输入流缓冲区的逗号与格式控制字符串中逗号刚好匹配,成功!

    法2(不够彻底):

    scanf("%d,%s ,%d", &a,mon,&b);   /*注意%s后面有个空格 */

    printf("%d,%s,%d\n",a,mon,b);

    并且在输入时加个空格

    12,fefe ,45   /*fefe和逗号之间加个空白*/

    相关知识:当scanf()格式控制字符串中出现空白时,表示取数时跳过任何空白。

    scanf读到fefe后的空格后结束字符串的读取,此时输入流缓冲区第一个字符为空格;继续读,由于格式控制字符串里有个空格,所以读取时会跳过任何空白(不信可以在fefe后面多敲几个空白试试,全都跳过,甚至连回车都跳过),读到逗号匹配成功。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值