C语言scanf函数奇遇记
作者:ocean 撰写日期:2011-11-20
博客链接:http://oceanspace.tk
看《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后面多敲几个空白试试,全都跳过,甚至连回车都跳过),读到逗号匹配成功。