感谢大佬:https://blog.csdn.net/qq_26768741/article/details/50933598

在C语言中,我们常常需要去清空缓存区,对于缓存区清空的重要性,接下来我们进行具体的说明。

首先我们先解释一下缓存区。

1.缓存区

C语言中的缓冲区又称为缓存,它是内存空间的一部分。

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

    C语缓冲区分为三种类型:1、全缓冲 2、行缓冲 3、不带缓冲。

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

例如,在我们平时要在磁盘中读取信息的情况下,先会把数据放到缓存区中,读取完后,再次从磁盘中读取信息。

缓存区,他的意义就是在高速CPU与低速的设备之间的一个区域,这个区域让CPU工作效率更高。

  1. 全缓冲

    当填满标准I/O缓存后才进行实际I/O操作。全缓冲的典型代表是对磁盘文件的读写。

  2. 行缓冲

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

  3. 不带缓冲

    也就是不进行缓冲,标准出错情况stderr是典型代表,这使得出错信息可以直接尽快地显示出来。

大部分系统默认使用下列类型的缓存:

标准出错是不带缓存的。

如果是涉及终端设备的流,则它们是行缓存的,否则是全缓存的。

我们经常用到的输入输出流,在目前的ANSI C 中缓存的特征是:stdin和stdout是行缓存;而stderr是无缓存的。

了解了这些,接下来我们进入正题,接下来我们要清楚为什么要清空缓存区,

这是因为在你有的时候在输入一个字符后,在输入一个字符,如果你不清空缓冲区,那上一个字符还在你的缓冲区内!这样就造成错误了!

2.清空缓存区

所以我们要对缓存区清空是很重要的,在这里我们给出了下面三种办法来实现:

  • 使用fflush()函数

    fflush()函数冲洗流中的信息,该函数通常用于处理磁盘文件。清除读写缓冲区,需要立即把输出缓冲区的数据进行物理写入时。fflush()函数包含在stdio.h头文件中。

    函数原:int fflush(FILE *stream),在这里的stream就是所要清除缓存区的文件。

    函数的返回值:当进行刷新成功返回0,失败返回EOF。没有缓冲区或者只读打开时也返回0值。还有需要注意的是:如果fflush返回EOF,数据可能由于写错误已经丢失。

    用法示例:fflush(stdin)刷新标准输入缓冲区,fflush(stdout)刷新标准输出缓冲区。 printf(“。。。。。。。。。。。”);后面加fflush(stdout);可提高打印效率

    代码示例:
#include<stdio.h>
#include<stdlib.h>
int main()
{
int num;
char a;
scanf("%d", &num);
a = getchar();
printf("num=%d,a=%c\n", num, a);
system("pause");
return 0;
}



在执行这个程序的时候我们会发现,在我们输入第一个字符后敲击‘\n’之后,程序就结束了,这是因为我们没有清空缓存区,getchar()函数接收的是‘\n’,最后造成了程序的停止。

接下来我们来使用fflush()函数。

#include<stdio.h>
#include<stdlib.h>
int main()
{
int num;
char a;
scanf("%d", &num);
fflush(stdin);
a = getchar();
printf("num=%d,a=%c\n", num, a);
system("pause");
return 0;
}



我们在这里可以看到,当使用了fflush()函数对输入流进行清空缓存区以后,就可以正常让getchar函数接收字符。

接下来是第二种方法。

  • 使用while((ch = getchar()) != ‘\n’ && ch != EOF);语句

    这种方法是最好的方法,可以称为万能清空缓存区御语句。

    代码示例:
#include <stdio.h>
#include<stdlib.h> int main()
{
char c1 = 0;
char c2 = 0;
scanf("%c", &c1);
c2 = getchar();
printf("c1=%d,c2=%c", c1, c2);
system("pause");
return 0;
}



在这里我们依然没有清空缓存区,得到的效果是这样的,getchar()函数依然接收了’\n’,造成程序停止。

接下来,我们使用while((ch = getchar()) != ‘\n’ && ch != EOF);语句

#include <stdio.h>
#include<stdlib.h> int main()
{
char c1 = 0;
char c2 = 0;
int ch;
scanf("%d", &c1);
while ((ch = getchar()) != EOF && ch != '\n')
{
;
}
c2 = getchar();
printf("c1=%d,c2=%c", c1, c2);
system("pause");
return 0;
}



加入了while ((ch = getchar()) != EOF && ch != ‘\n’);语句,完成了清空缓存区

事实上有时我们会将这个语句封装在一个函数中,这样会让程序的可移植性更强。

#define CLEAR_BUF \
{\
int ch; \
while ((ch = getchar()) != EOF && ch != '\n')\
{\
; \
}\
} #include <stdio.h>
#include<stdlib.h> int main()
{
char c1 = 0;
char c2 = 0;
scanf("%d", &c1);
CLEAR_BUF
c2 = getchar();
printf("c1=%d,c2=%c", c1, c2);
system("pause");
return 0;
}



依然可以达到我们要的效果。

  • setbuf()函数关闭缓存区

接下来我们要说另外一种清空缓存区的方法,与其说是清空缓存区,不如说他是关闭缓存区。

setbuf()函数

是linux中的C函数,主要用于打开和关闭缓冲机制。包含在头文件stdio.h中。

setbuf函数具有打开和关闭缓冲机制。为了带缓冲进行I/O,参数buf必须指向一个长度为BUFSIZ的缓冲区。通常在此之后该流就是全缓冲的,但是如果该流与一个终端设备相关,那么某些系统也可以将其设置为行缓冲。为了关闭缓冲,可以将buf参数设置为NULL。

函数原型:void setbuf(FILE *stream,char *buf);一个参数是文件流,一个参数是buf指向的缓冲区长度,这个长度就是在stdio.h中定义的宏BUFSIZ所决定的。当定义buf为空时,setbuf函数将使的文件I/O不带缓冲。

#include <stdio.h>
#include<stdlib.h>
char outbuf[50];
int main(void)
{
/* 将outbuf与stdout输出流相连接 */
setbuf(stdout, outbuf);
/* 向stdout中放入一些字符串 */
puts("This is a test of buffered output.");
puts("This output will go into outbuf");
puts("and won't appear until the buffer");
puts("fills up or we flush the stream.\n");
/* 以下是outbuf中的内容 */
puts(outbuf);
/*刷新流*/
fflush(stdout);
system("pause");
return 0;
}

C语言之清空缓存区的更多相关文章

  1. 基础 - 字符读取函数scanf、getchar、gets、cin(清空缓存区解决单字符回车问题)

    0x01 scanf.getchar.cin读取单字符: 如下: //scanf读取字符 回车问题 void Sub_1_1() { char v1,v2; scanf("%c", ...

  2. C语言 标准输入 清空缓存

  3. sed原理及sed命令格式 ,缓存区,模式空间

    4.1            Sed工作原理 sed是一个非交互式的流编辑器.所谓非交互式,是指使用sed只能在命令行下输入编辑命令来编辑文本,然后在屏幕上查看输出:而所谓流编辑器,是指sed每次只从 ...

  4. 【MINA】缓存区ByteBuffer和IOBuffer你要了解的常用知识

    mina中IOBuffer是Nio中ByteBuffer的衍生类,主要是解决Bytebuffer的两个不足 1.没有提供足够灵活的get/putXXX方法 2.它容量固定,难以写入可变长度的数据 特点 ...

  5. JAVA之旅(十七)——StringBuffer的概述,存储,删除,获取,修改,反转,将缓存区的数据存储到数组中,StringBuilder

    JAVA之旅(十七)--StringBuffer的概述,存储,删除,获取,修改,反转,将缓存区的数据存储到数组中,StringBuilder 讲完String,我们来聊聊他的小兄弟 一.StringB ...

  6. 关于cin的缓存区影响效果

    1. 当缓冲区中有残留数据时,cin函数会直接去读取这些残留数据而不会请求键盘输入.而且,回车符也会被存入输入缓冲区中. int num{}; while(cin>>num) cout&l ...

  7. 一个fork的面试题——fork + 缓存区

    update : 20140512 题目:请问下面的程序一共输出多少个“-”? #include <stdio.h> #include <sys/types.h> #inclu ...

  8. 进程通过内核缓存区请求设备I/O的一些事情

    请求进程,内核缓存区,设备I/O 请求进程无法直接访问设备I/O,而是通过内核缓冲区提交请求数据,等数据就绪后,数据从设备缓冲区提交至进程空间 请求进程把数据提交给内核缓存空间需要等待,内核把数据复制 ...

  9. node(Buffer缓存区)

    // 创建buffer类 var buf=new buffer(10); var buf=new buffer([10,20,30,40]); var buf=new buffer("www ...

随机推荐

  1. # 中文NER的那些事儿6. NER新范式!你问我答之MRC

    就像Transformer带火了"XX is all you need"的论文起名大法,最近也看到了好多"Unified XX Framework for XX" ...

  2. 使用 DDL 语句分别创建仓库表、供应商表、产品表和入库表,并对其进行操作

    查看本章节 查看作业目录 需求说明: 使用 DDL 语句分别创建仓库表.供应商表.产品表和入库表 使用 DML 语句分别新增一条仓库表.供应商表.产品表和入库表记录 使用 DML 语句更新满足指定条件 ...

  3. Java初学者作业——定义管理员类(Admin),管理员类中的属性包括:姓名、账号、密码、电话;方法包括:登录、显示自己的信息。

    返回本章节 返回作业目录 需求说明: 定义管理员类(Admin),管理员类中的属性包括:姓名.账号.密码.电话:方法包括:登录.显示自己的信息. 实现思路: 分析类的属性及其变量类型. 分析类的方法及 ...

  4. .NetCore下构建自己的服务配置中心-手动造轮子

    本人主要利用IdentityServer4以及SignalR来实现,IdentityServer4作为认证,SignalR来交互配置,这里一些代码可能就是部分提出来,主要介绍实现原理及方法 实现配置中 ...

  5. 初识python 之 爬虫:使用正则表达式爬取“古诗文”网页数据

    通过requests.re(正则表达式) 爬取"古诗文"网页数据. 详细代码如下: #!/user/bin env python # author:Simple-Sir # tim ...

  6. 基于appnium+python+夜神模拟器的自动化

    首先搭好appnium环境!参考https://www.cnblogs.com/testlearn/p/11419797.html 1.安装夜神模拟器 下载安装夜神模拟器后,在cmd命令输入adb c ...

  7. linux安全篇:禁止频繁访问的ip访问nginx

    实验环境 版本:redhat6.5ip:172.16.1.100,172.16.10软件:nginx 172.16.1.10部署nginx [root@localhost tools]# lsngin ...

  8. Linux sudo 找不到命令

    普通用户执行需要root权限的命令,提示"找不到命令",但是root用户执行该命令不报错,可能是由于该命令未处在sudo搜索的路径. 本文以sudo easy_install 为例 ...

  9. sqlserver - 查出的结果集,集成为json串放在一个字段里

    1.效果 2.sql SELECT top 20 (select [name] as [名字] from staffBasicInfo For JSON PATH,ROOT('第一级key')) k ...

  10. dispatcher-servlet.xml文件配置模板

    完整代码如下: <?xml version="1.0" encoding="UTF-8"?><beans xmlns="http:/ ...