当前位置:编程学习 > C/C++ >>

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。www.zzzyk.com
    那有什么办法实现用逗号作为间隔符的情况呢?下面提供两种方法:
    五、解决方法
    法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后面多敲几个空白试试,全都跳过,甚至连回车都跳过),读到逗号匹配成功。 

摘自  家里蹲博客 
补充:软件开发 , C语言 ,
CopyRight © 2022 站长资源库 编程知识问答 zzzyk.com All Rights Reserved
部分文章来自网络,