scanf函数返回值
-
正整数,表示正确输入参数的个数。例如执行 scanf(“%d %d”, &a, &b);
如果用户输入”3 4”,可以正确输入,返回2(正确输入了两个变量);
如果用户输入”3,4”,可以正确输入a,无法输入b,返回1(正确输入了一个变量)。
- 0,表示用户的输入不匹配,无法正确输入任何值。如上例,用户如果输入”,3 4”,返回0。
- EOF,这是在stdio.h里面定义的常量(通常值为-1),表示输入流已经结束。在Windows下,用户按下CTRL+Z(会看到一个^Z字符)再按下回车(可能需要重复2次),就表示输入结束;Linux/Unix下使用CTRL+D表示输入结束。
关于缓冲区问题
-
缓冲区又称为缓存,它是内存空间的一部分。也就是说,在内存空间中预留了一定的存储空间,这些存储空间用来缓冲输入或输出的数据,这部分预留的空间就叫做缓冲区。
缓冲区根据其对应的是输入设备还是输出设备,分为输入缓冲区和输出缓冲区。
是一块内存区,它用在输入输出设备和CPU之间,用来缓存数据。它使得低速的输入输出设备和高速的CPU能够协调工作,避免低速的输入输出设备占用CPU,解放出CPU,使其能够高效率工作。
缓冲区的类型
缓冲区 分为三种类型:全缓冲、行缓冲和不带缓冲。
- 全缓冲 在这种情况下,当填满标准I/O缓存后才进行实际I/O操作。全缓冲的典型代表是对磁盘文件的读写。
-
行缓冲
在这种情况下,当在输入和输出中遇到换行符时,执行真正的I/O操作。这时,我们输入的字符先存放在缓冲区,等按下回车键换行时才进行实际的I/O操作。典型代表是标准输入(stdin)和标准输出(stdout)。
- 不带缓冲 也就是不进行缓冲,标准出错情况stderr是典型代表,这使得出错信息可以直接尽快地显示出来。
我们经常要用到标准输入输出流,而ANSI
C对stdin、stdout和stderr的缓存特征没有强行的规定,以至于不同的系统可能有不同的stdin、stdout和stderr的缓存特征。目前主要的缓存特征是:stdin和stdout是行缓存;而stderr是无缓存的。
缓冲区的大小
如果我们没有自己设置缓冲区的话,系统会默认为标准输入输出设置一个缓冲区,这个缓冲区的大小通常是512个字节的大小。
缓冲区大小由 stdio.h 头文件中的宏 BUFSIZ 定义,如果希望查看它的大小,包含头文件,直接输出它的值即可:
printf(“%d”, BUFSIZ);
缓冲区的大小是可以改变的,也可以将文件关联到自定义的缓冲区,详情可以查看 setvbuf() 和 setbuf() 函数。
缓冲区的刷新(清空)
下列情况会引发缓冲区的刷新:
1、缓冲区满时;
2、行缓冲区遇到回车时;
3、关闭文件;
4、使用特定函数刷新缓冲区。
-
c语言中在读取键盘数据时,一般是带缓存的数据输入,需要按回车键才能完成该“行”数据的输入确认。
而 scanf()函数对这个回车确认符并不进行处理,回车符会留在输入缓存区中。
因此,在下一个读“字符”操作函数(getchar, scanf("%c"), gets()等)运行时,会读到这个字符。
而在读数值型数据或字符串时,scanf()会从第一个非空白字符(空白字符指:回车,空格,TAB键)开始读取,自动忽略前面的空白字符,而遇到空白字符结束该类型数据的输入。
因此,是否一定要在scanf后面跟个 getchar吃回车,要看下一个输入的数据类型是什么,如果是读字符类操作,处理办法有多种方式:
- 用fflush(stdin)命令强行刷新输入缓存,丢弃输入缓存中的数据,此种方法在windows下使用有效,linux无效
char ch;
fflush(stdin); //不管缓存中有没有数据,强行清除
ch=getchar(); //这里会等待用户输入一个字符
- 前面有读数据操作,现在要执行读字符操作,则可用getchar()来吃掉前面的回车确认
int i;
char ch;
scanf("%d", &i ); //读一个整数,回车确认
getchar(); //先吃掉回车确认符,这两句,效果同下一方法的那一句
ch=getchar(); //这里会等待用户输入一个字符
- 在scanf()书写上做文章,如:读一个数据,以回车确认,则输写成如下格式
int i;
char ch;
scanf("%d%*c", &i ); //%*c表示读一个字符,并不赋值给任何变量
ch=getchar(); //这里不会读到回车符
其他:
scanf接收包含空格的字符串
char str[80];
scanf("%[^\n]",str); //读到'\n'结束读取
//读到'\n'结束读取,存入str,再抛弃一个字符
scanf("%[^\n]%*c",str);
//读到'\n'结束读取,并将其读到的数据抛弃,然后再抛弃一个字符(这个字符是'\n')
//此时缓存中不存在任何字符
scanf("%*[^\n]%*c");
int c;
while((c=getchar())!='\n'&&c!=EOF);
//读取一个字符,直到是\n或者是EOF停止
//等价于
scanf("*[^\n]");
//直到遇到字符a停止读取,
//舍弃a
//读取将a后的一个字符存入变量c
scanf("%*[^a]%*c%c",&c);