scanf格式化中的\n
如果一个scanf的格式串以\n结尾,那么在读取完后还会阻塞等待,比如:
1 int a;
2 scanf("%d\n", &a);
这种情况,输入一个数字然后敲下回车后,程序还是阻塞着的。为什么呢?
处理格式串中的普通字符时,scanf函数采取的动作依赖于这个字符是否为空白字符(回车、tab、空格)。
如果是空白字符。当在格式串中遇到一个或多个连续的空白字符时,scanf函数从输入中重复读空白字符直到遇到一个非空白字符(把该字符“放回原处”)为止。格式串中的一个空白字符可以与输入中任意数量的空白字符相匹配,包括0个。
scanf("%d\n", &a)中的\n不表示等待换行符,而是读取并放弃连续的空白字符,你输入一个数之后,它是不会立即显示的,要等再接收到一个非空白字符(空格、制表符、回车)的输入scanf语句才结束。但请注意的是,最后输入的那个非(空格、制表符、回车)的东西是不会被这个scanf读进来的,而是留在输入流里。
格式串中的’\n’、‘tab’、‘空格’都属于空白字符,因此当遇到‘\n’时,程序会重复读空白字符直到遇到一个非空白字符为止”,由于输入缓冲已经没有字符可读了,因此将阻塞等待,直到读入了一个非空白字符为止。当程序阻塞的时候,我们直接按下回车,程序还是继续阻塞,只有当我们输入一个非空白字符并按下回车时程序才能继续向下执行。
如果格式串是scanf("%c",…),那么%c会将空白字符读入并且成功返回。
scanf读取数值类型时用若用回车结束输入,那么会在缓冲区中留下’\n’,当下面紧接着一个读取字符类型的函数(比如scanf("%c",b),或者gets())时,’\n’将会被读取,造成程序错误,这个问题必须引起注意。如果第一个scanf读取的是数值类型,例如scanf("%d", &num),而紧接着的scanf也是读取的数值类型,则缓冲区的回车会被忽略,可以成功读取后面的数值。因为,缓冲区中的回车是属于字符,只有读取字符时才会把它当成一个有效的输入。