天天看点

C语言scanf()与缓冲区

scanf函数返回值

  1. 正整数,表示正确输入参数的个数。例如执行 scanf(“%d %d”, &a, &b);

    如果用户输入”3 4”,可以正确输入,返回2(正确输入了两个变量);

    如果用户输入”3,4”,可以正确输入a,无法输入b,返回1(正确输入了一个变量)。

  2. 0,表示用户的输入不匹配,无法正确输入任何值。如上例,用户如果输入”,3 4”,返回0。
  3. EOF,这是在stdio.h里面定义的常量(通常值为-1),表示输入流已经结束。在Windows下,用户按下CTRL+Z(会看到一个^Z字符)再按下回车(可能需要重复2次),就表示输入结束;Linux/Unix下使用CTRL+D表示输入结束。

关于缓冲区问题

  • 缓冲区又称为缓存,它是内存空间的一部分。也就是说,在内存空间中预留了一定的存储空间,这些存储空间用来缓冲输入或输出的数据,这部分预留的空间就叫做缓冲区。

    缓冲区根据其对应的是输入设备还是输出设备,分为输入缓冲区和输出缓冲区。

    是一块内存区,它用在输入输出设备和CPU之间,用来缓存数据。它使得低速的输入输出设备和高速的CPU能够协调工作,避免低速的输入输出设备占用CPU,解放出CPU,使其能够高效率工作。

    缓冲区的类型

    缓冲区 分为三种类型:全缓冲、行缓冲和不带缓冲。

    1. 全缓冲 在这种情况下,当填满标准I/O缓存后才进行实际I/O操作。全缓冲的典型代表是对磁盘文件的读写。
    2. 行缓冲

      在这种情况下,当在输入和输出中遇到换行符时,执行真正的I/O操作。这时,我们输入的字符先存放在缓冲区,等按下回车键换行时才进行实际的I/O操作。典型代表是标准输入(stdin)和标准输出(stdout)。

    3. 不带缓冲 也就是不进行缓冲,标准出错情况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吃回车,要看下一个输入的数据类型是什么,如果是读字符类操作,处理办法有多种方式:

  1. 用fflush(stdin)命令强行刷新输入缓存,丢弃输入缓存中的数据,此种方法在windows下使用有效,linux无效
char ch;
fflush(stdin); //不管缓存中有没有数据,强行清除
ch=getchar(); //这里会等待用户输入一个字符
           
  1. 前面有读数据操作,现在要执行读字符操作,则可用getchar()来吃掉前面的回车确认
int i;
char ch;
scanf("%d", &i ); //读一个整数,回车确认
getchar(); //先吃掉回车确认符,这两句,效果同下一方法的那一句
ch=getchar(); //这里会等待用户输入一个字符
           
  1. 在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);
           

继续阅读