1、scanf()工作原理
scanf()根据一个%d读取一个数值,每次读取前都跳过数值前的所有空白字符(空格、tab、换行符),直至遇到非空白字符才开始读取。因为要读取整数,当scanf()找到数字或者+-号时,会把它保存下来并继续读取下一个数字,如果数字是连续的就一直读取并保存。如果遇到空白或者其他非数字字符,scanf()会认为读到了整数的末尾,然后把非数字字符放回输入,这意味着程序在下次读取输入时首先读取到的是上一次被放回输入的非数字字符。
2、详细的输入过程
写一条C语句:scanf("%d%d%d", &a, &b, &c); 输入下面内容
1 22
3
首先scanf()读取第一个%d,读取到1然后遇到空格,读取结束并把空格放回输入里。接着开始读取第二个%d,读取前跳过输入里的全部空格(3个),读取到2和2并保存,遇到换行符读取结束并把换行符放回输入,读取第三个%d,跳过被放回输入里的换行符,读取3,最后读取结束并把换行符放回输入。如果这个程序下面还要scanf(%d),留在输入里的这个换行符会被跳过,如图1。但如果用用scanf(%c),这个换行符就会被读取到并赋值给对应的变量。如下面图2
3、%d读取到非数字字符
如果%d读取到的第一个不是数字而是字母a,scanf()会停在那里,并把字母a留在输入缓冲区中,不会把值赋给指定变量,然后程序在下一次读取输入缓冲区时首先读取到的还是a。如果程序之后还是只使用%d来读取,那么scanf()就一直无法读取整数并赋值,a会一直留在在输入缓冲区中。另外,如果使用多个带转换说明的scanf(),scanf()会在第一个出错的地方停止读取输入。
例如下面这段程序,输入字母d,scanf()不会给变量a赋值(a还是原来的值),而会把字母d放回输入缓冲区,在下一次%c读取的时候把字符d赋值给变量b。
最后:这篇笔记是在C primer plus的基础上加上自己的理解写出来的,可能会有不准确的地方,如果有错希望大佬可以指正。
参考资料:C primer plus(第6版)