参考资料:

  https://www.douban.com/group/topic/127062773/

  今天学习了C++语言的标准I/O,也就是std::cin和std::cout,但是我发现当系统在读取标准的输入后需要按两次ctrl+D或者按一次回车再按一次ctrl+D才能结束标准I/O,翻阅相关资料后我把这个问题研究透彻了记录在此。(使用类Unix系统,所以EOF是ctrl+D,windows上可能是ctrl+Z)

  首先我们必须知道一个概念:缓冲区,缓冲区是干什么的?


  我们为什么要引入缓冲区?

  比如我们从磁盘里取信息,我们先把读出的数据放在缓冲区,计算机再直接从缓冲区中取数据,等缓冲区的数据取完后再去磁盘中读取,这样就可以减少磁盘的读写次数,再加上计算机对缓冲区的操作大大快于对磁盘的操作,故应用缓冲区可大大提高计算机的运行速度。

  又比如,我们使用打印机打印文档,由于打印机的打印速度相对较慢,我们先把文档输出到打印机相应的缓冲区,打印机再自行逐步打印,这时我们的CPU可以处理别的事情。现在您基本明白了吧,缓冲区就是一块内存区,它用在输入输出设备和CPU之间,用来缓存数据。它使得低速的输入输出设备和高速的CPU能够协调工作,避免低速的输入输出设备占用CPU,解放出CPU,使其能够高效率工作。  


  缓冲区的种类?

  1. 全缓冲:在这种情况下,当填满标准I/O缓存后才进行实际I/O操作。全缓冲的典型代表是对磁盘文件的读写。
  2. 行缓冲:在这种情况下,当在输入和输出中遇到换行符时,执行真正的I/O操作。这时,我们输入的字符先存放在缓冲区,等按下回车键换行时才进行实际的I/O操作。典型代表是键盘输入数据。
  3. 不带缓冲:也就是不进行缓冲,标准出错情况stderr是典型代表,这使得出错信息可以直接尽快地显示出来。

  我们研究的对象是键盘的输入,也就是行缓冲了,下面用一个例子说明问题:

#include <iostream>

/*
* Simple main function:
* Read several numbers and write their sum
*/
int main()
{
int sum = 0, val = 1;
while(std::cin >> val){
sum += val;
}
std::cout << "The sum is " << sum << std::endl;
return 0;
}

  在这个例子中我们从标准输入中取值,赋给val,当赋给val的值是可用的值时,while循环继续,否则打破循环。

  读者可以试验一下这两种情况:

  1.运行程序之后输入1 2 3然后按回车键,此时程序不结束,再按ctrl+D键,也就是输入EOF,程序给出输出6。

  2.运行程序之后输入1 2 3 (注意3后面我多输入一个空格,上面那个例子我不输入)然后按ctrl+D,没反应,再按ctrl+D,程序输出6。

  我们用学到的缓冲区的知识细致细致再细致地分析这个过程:

  第一种情况:

  输入1 2 3,此时缓冲区是这样的:'1' '空格' '2' '空格' '3',因为行缓冲,输入回车后'1' '空格' '2' '空格' '3' '回车'被送给CPU,也就是送给while了,while拿到数据后很高兴,按照空格和回车都是分割符的原理,将1 2 3交给val加了起来。此时缓冲区呢:空了!!但是程序还在期待输入,但我不想再输入了,那么在一个空的缓冲区中输入ctrl+D就可以直接结束标准输入。

  第二种情况:

  输入1 2 3 (3后面有空格),此时缓冲区是这样的:'1' '空格' '2' '空格' '3' '空格',根据EOF符号的定义,此时我们打出EOF符号(ctrl+D)。'1' '空格' '2' '空格' '3' '空格'被送给CPU,同时EOF被丢弃了。while拿到数据后又很高兴,按照空格是分割符的原理,将1 2 3交给val加了起来。但是输入还没完啊,此时'1' '空格' '2' '空格' '3' '空格'交给CPU了,EOF被丢弃了,缓冲区是不是空了?此时再打出EOF符号,标准输入即被关闭。

  读者可能会问为什么第二种情况在3后面要加上空格,是这样的,可以考虑不加空格是什么样的结果:CPU会拿到'1' '空格' '2' '空格' '3',此时因为3后面没有分隔符,那么我们此时输入EOF就是把标准输入截断了。假如我们输入'8' '空格',再按两次ctrl+D,此时sum的加和是1+2+38=41,所以说这种情况就和std::cin传值机理的问题了,和缓冲区的原理关系不大。

  总结一下:

  回车键的效果:将缓冲区连带自身传给CPU,缓冲区清空。

  ctrl+D(EOF)的效果:将缓冲区不带自身传给CPU,缓冲区清空。

  缓冲区为空时收到EOF,标准文件输入关闭!

C++中为什么按两次ctrl+D才能结束标准I/O的更多相关文章

  1. 浏览器缓存如何控制? && 在url框中回车、F5 和 Ctrl + F5的区别是什么?

    第一部分: 浏览器缓存如何控制?   最近在做网站,但是不知道缓存是什么东西怎么能行! 如何实现HTTP缓存呢? 下面我们来一步一步的探寻实现机制把. 方案一: 无缓存   说明: 浏览器向服务器请求 ...

  2. 浏览器缓存控制 以及 在url框中回车、F5 和 Ctrl + F5的区别

    第一部分: 浏览器缓存如何控制?   做网站,不知道缓存是什么东西怎么能行! 如何实现HTTP缓存呢? 下面我们来一步一步的探寻实现机制把. 方案一: 无缓存   说明:浏览器向服务器请求资源m.pn ...

  3. Android中Fragment的两种创建方式

    fragment是Activity中用户界面的一个行为或者是一部分.你可以在一个单独的Activity上把多个Fragment组合成为一个多区域的UI,并且可以在多个Activity中再使用.你可以认 ...

  4. 【跟着子迟品 underscore】JavaScript 中如何判断两个元素是否 "相同"

    Why underscore 最近开始看 underscore.js 源码,并将 underscore.js 源码解读 放在了我的 2016 计划中. 阅读一些著名框架类库的源码,就好像和一个个大师对 ...

  5. [Linux]在linux中,常常用到ctrl和其他按键组合,常用的有哪些及意义呢

    在linux中,常常用到ctrl和其他按键组合,常用的有哪些及意义呢? Ctrl+c 结束正在运行的程序 Ctrl+d 结束输入或退出shell Ctrl+s 暂停屏幕输出[锁住终端] Ctrl+q ...

  6. java 中 return 的两种常见的用法

    一:return语句总是用在方法中,有两个作用: 一个是返回方法指定类型的值(这个值总是确定的), 一个是结束方法的执行(仅仅一个return语句). 二:实例1 -- 返回一个String priv ...

  7. java中线程分两种,守护线程和用户线程。

    java中线程分为两种类型:用户线程和守护线程. 通过Thread.setDaemon(false)设置为用户线程: 通过Thread.setDaemon(true)设置为守护线程. 如果不设置次属性 ...

  8. Android向系统相册中插入图片,相册中会出现两张 一样的图片(只是图片大小不一致)

    向系统相册中插入图片调用此方法时,相册中会出现两张一样的图片 MediaStore.Images.Media.insertImage 一张图片是原图一张图片是缩略图.表现形式为:android4.4. ...

  9. 处理Selection对象和Range对象——Word VBA中重要的两个对象

    处理Selection对象和Range对象——Word VBA中重要的两个对象 Word 开发人员参考Selection 对象代表窗口或窗格中的当前所选内容.所选内容代表文档中选定(或突出显示)的区域 ...

随机推荐

  1. MySQL事务锁等待超时 Lock wait timeout exceeded; try restarting transaction

    工作中处理定时任务分发消息时出现的问题,在查找并解决问题的时候,将相关的问题博客收集整理,在此记录下,以便之后再遇到相同的问题,方便查阅. 问题场景 问题出现的场景: 在消息队列处理消息时,同一事务内 ...

  2. 破解webstorm 亲测有效

    一.首先安装好webstorm,并且配置hosts文件 二.使用以下激活码进行激活,亲测有效可以用到2099年 4RULSIH54N-eyJsaWNlbnNlSWQiOiI0UlVMU0lINTROI ...

  3. vue中生命周期

    1,说器生命周期,总觉得有熟悉,又陌生,直到看到一道面试题,问父子组件的生命周期的执行顺序,我擦,真没太注意啊,不知道. 2,网上搜了一下,说法是有点像洋葱圈的形式,由外到内,在到外,因为就像一个盒子 ...

  4. [Objective-C] 005_Category(类别)

    Category的实际作用就是为已有的类来添加方法.为现有的类添加的方法可以先不用实现,在需要的时候再实现也是可以的.在我们的实际代码中如何来实现Category的呢?我们上篇的Person 类为例. ...

  5. 使用fileupload组件

    1. 进行文件上传时, 表单需要做的准备: 1). 请求方式为 POST: <form action="uploadServlet" method="post&qu ...

  6. 虚拟机安装中标麒麟桌面版7.0系统 + 升级Firefox浏览器

    背景 由于公司业务(政府项目)需要走国产化路线,需要把原来已有的产品在国产的系统进行测试.目前选择的是中标麒麟系统,这是一款国产系统,界面 UI 和 window 类似,系统内核使用的是 Linux ...

  7. 前端基础知识之html和css全解

    前端回顾 目录 前端回顾 基础知识 HTTP协议 认识HTML HTML组成 HTML标签 div和span标签 特殊的属性 常用标签 认识css 选择器 属性 前端就是展示给用户并且与用户进行交互的 ...

  8. 15 . PythonWeb框架本质

    PythonWeb框架的本质 简单描述就是:浏览器通过你输入的网址给你的socket服务端发送请求,服务端接受到请求给其回复一个对应的html页面,这就是web项目.所有的Web应用本质上就是一个so ...

  9. 基于 abp vNext 和 .NET Core 开发博客项目 - 定时任务最佳实战(三)

    上一篇(https://www.cnblogs.com/meowv/p/12974439.html)完成了全网各大平台的热点新闻数据的抓取,本篇继续围绕抓取完成后的操作做一个提醒.当每次抓取完数据后, ...

  10. 记录B端和C端产品的理解

    C 为:Consumer.Client,我们每天都在接触C端产品,为消费者.个人用户或终端用户,比如:微信.头条.抖音.美团等等. B 为:Business,作为职场人士也会经常接触B端产品,通常为企 ...